Testing Pagination with Selenium
last modified March 22, 2025
This tutorial provides a detailed walkthrough for constructing a Flask application featuring pagination, integrated with an SQLite database. It includes generating 100 sample data rows and employs Selenium for automated testing of the pagination functionality. The application adheres to Flask's recommended application factory pattern for modularity and scalability.
Project Structure
The following describes the structure of the application.
flask_pagination/ ├── instance/ │ └── example.db (SQLite database, created at runtime) ├── flask_pagination/ │ ├── __init__.py (Application factory) │ ├── db.py (Database initialization and utilities) │ ├── routes.py (Route definitions) │ └── templates/ │ └── index.html (HTML template) ├── tests/ │ └── test_app.py (Selenium tests) ├── run.py (Entry point to run the app) └── requirements.txt
Set Up the Flask Application
The application uses the factory pattern (create_app
) for
modularity and stores an SQLite database in the instance/
directory
for isolation. It implements pagination to display 10 items per page from 100
total rows, and includes CLI commands (init-db
and
populate-db
) for setting up the database. The database is
initialized with an items
table and populated with 100 rows labeled
'Item 1' to 'Item 100.' Selenium tests configure a test app and database,
confirm pagination functionality across various pages, and clean up temporary
files after testing.
Entry Point
This script serves as the entry point to launch the Flask application, utilizing the application factory pattern for instantiation.
from flask_pagination import create_app if __name__ == '__main__': app = create_app() app.run(debug=True)
The run.py
file is the primary executable for starting the Flask
application. It imports the create_app
function from the
flask_pagination package and invokes it to instantiate the
application object. The app.run(debug=True)
command launches the development server with debugging enabled, facilitating
real-time error tracking and automatic reloading during development.
Application Factory
This module establishes the application factory, configuring essential settings and integrating blueprints and database utilities.
from flask import Flask import click import os def create_app(test_config=None): # Create and configure the app app = Flask(__name__, instance_relative_config=True) # Ensure the instance folder exists try: os.makedirs(app.instance_path) except OSError: pass # Load configuration if test_config is None: app.config.from_mapping( DATABASE=os.path.join(app.instance_path, 'example.db'), ) else: app.config.from_mapping(test_config) # Register database commands from . import db db.init_app(app) # Register routes from . import routes app.register_blueprint(routes.bp) return app
The __init__.py
file defines the create_app
function,
central to Flask's application factory pattern. It initializes a Flask instance
with a relative instance path for configuration files and ensures the instance
directory exists using os.makedirs
.
Configuration is set dynamically: the default SQLite database path is assigned
unless overridden by a test configuration. The function integrates database
utilities via db.init_app
and registers the routes blueprint,
returning a fully configured application instance.
Database Handling
This module oversees SQLite database connectivity, schema creation, and data population with custom CLI commands.
import sqlite3 import click from flask import current_app, g def get_db(): if 'db' not in g: g.db = sqlite3.connect( current_app.config['DATABASE'], detect_types=sqlite3.PARSE_DECLTYPES ) g.db.row_factory = sqlite3.Row return g.db def close_db(e=None): db = g.pop('db', None) if db is not None: db.close() def init_db(): db = get_db() db.execute('DROP TABLE IF EXISTS items') db.execute('CREATE TABLE items (id INTEGER PRIMARY KEY, name TEXT)') db.commit() def populate_db(): db = get_db() for i in range(1, 101): db.execute('INSERT OR IGNORE INTO items (id, name) VALUES (?, ?)', (i, f'Item {i}')) db.commit() def init_app(app): app.teardown_appcontext(close_db) app.cli.add_command(init_db_command) app.cli.add_command(populate_db_command) @click.command('init-db') def init_db_command(): """Initialize the database.""" init_db() click.echo('Initialized the database.') @click.command('populate-db') def populate_db_command(): """Populate the database with 100 sample items.""" populate_db() click.echo('Database populated with 100 items.')
The db.py
module manages database operations for the Flask application. The
get_db
function establishes a connection to the SQLite database,
storing it in Flask's g
object for reuse within a request context,
with rows returned as dictionary-like objects. close_db
ensures
connections are closed post-request.
The init_db
function creates a fresh items
table,
dropping any existing one, while populate_db
inserts 100 rows
(e.g., 'Item 1' to 'Item 100'). The init_app
function integrates
these with the app, adding CLI commands (init-db
and
populate-db
) for database management, enhancing usability.
Routes
This module implements the application's routing and pagination logic using a Flask Blueprint for modularity.
from flask import Blueprint, render_template, request from .db import get_db import math bp = Blueprint('main', __name__) def get_items(page, per_page=10): offset = (page - 1) * per_page db = get_db() items = db.execute('SELECT * FROM items LIMIT ? OFFSET ?', (per_page, offset)).fetchall() total_items = db.execute('SELECT COUNT(*) FROM items').fetchone()[0] return items, total_items @bp.route('/') def index(): page = request.args.get('page', 1, type=int) per_page = 10 items, total_items = get_items(page, per_page) total_pages = math.ceil(total_items / per_page) return render_template('index.html', items=items, page=page, total_pages=total_pages, per_page=per_page)
The routes.py
module defines the application's routing logic within
a Blueprint named main
. The get_items
function
retrieves a page of items from the database, calculating an offset based on the
page number and items per page (defaulting to 10).
It uses SQL LIMIT
and OFFSET
for pagination and
returns the items alongside the total count. The index
route
handles the root URL, extracting the requested page from query parameters,
fetching the corresponding items, and rendering the index.html
template with pagination data, including current page, total pages, and items
per page.
Template
This template renders a paginated table of items with navigation controls.
<!DOCTYPE html> <html> <head> <title>Pagination Example</title> <style> .pagination { margin: 20px 0; } .pagination a { padding: 8px 16px; text-decoration: none; color: black; } .pagination a.active { background-color: #4CAF50; color: white; } </style> </head> <body> <h1>Items List</h1> <table> <tr> <th>ID</th> <th>Name</th> </tr> {% for item in items %} <tr> <td>{{ item['id'] }}</td> <td>{{ item['name'] }}</td> </tr> {% endfor %} </table> <div class="pagination"> {% if page > 1 %} <a href="?page={{ page - 1 }}">« Previous</a> {% endif %} {% for p in range(1, total_pages + 1) %} <a href="?page={{ p }}" class="{% if p == page %}active{% endif %}">{{ p }}</a> {% endfor %} {% if page < total_pages %} <a href="?page={{ page + 1 }}">Next »</a> {% endif %} </div> </body> </html>
The index.html
template presents a table of paginated items and
navigation links. It includes inline CSS to style pagination controls,
highlighting the active page in green. The table displays item IDs and names
dynamically using Jinja2's for
loop over the items
list. Pagination links include 'Previous' and 'Next' buttons, conditionally
shown based on the current page, and numbered links for each page, generated via
a range
loop. This structure ensures an intuitive user interface
for navigating the dataset.
Selenium Unit Tests
This module contains Selenium-based unit tests to validate the pagination functionality of the Flask application.
import unittest from selenium import webdriver from selenium.webdriver.common.by import By from selenium.webdriver.chrome.service import Service from webdriver_manager.chrome import ChromeDriverManager import time import os from flask_pagination import create_app class TestPagination(unittest.TestCase): def setUp(self): # Set up Flask app with test configuration including DATABASE self.app = create_app({ 'TESTING': True, 'DATABASE': os.path.join(os.path.abspath(os.path.dirname(__file__)), '..', 'instance', 'test_example.db') }) self.client = self.app.test_client() # Initialize and populate database within app context with self.app.app_context(): from flask_pagination.db import init_db, populate_db init_db() populate_db() # Start Flask server in a separate thread import threading self.server_thread = threading.Thread(target=self.app.run, kwargs={'port': 5000}) self.server_thread.daemon = True self.server_thread.start() time.sleep(1) # Give server time to start # Set up Selenium self.driver = webdriver.Chrome(service=Service(ChromeDriverManager().install())) self.driver.get('http://localhost:5000') time.sleep(1) # Wait for page to load def tearDown(self): self.driver.quit() # Clean up test database with self.app.app_context(): db_path = self.app.config['DATABASE'] if os.path.exists(db_path): os.remove(db_path) def test_initial_page_load(self): rows = self.driver.find_elements(By.XPATH, '//table//tr[td]') self.assertEqual(len(rows), 10) first_item = self.driver.find_element(By.XPATH, '//table//tr[td][1]/td[2]').text last_item = self.driver.find_element(By.XPATH, '//table//tr[td][10]/td[2]').text self.assertEqual(first_item, 'Item 1') self.assertEqual(last_item, 'Item 10') def test_pagination_next(self): next_button = self.driver.find_element(By.LINK_TEXT, 'Next »') next_button.click() time.sleep(1) rows = self.driver.find_elements(By.XPATH, '//table//tr[td]') self.assertEqual(len(rows), 10) first_item = self.driver.find_element(By.XPATH, '//table//tr[td][1]/td[2]').text last_item = self.driver.find_element(By.XPATH, '//table//tr[td][10]/td[2]').text self.assertEqual(first_item, 'Item 11') self.assertEqual(last_item, 'Item 20') def test_pagination_specific_page(self): page_5 = self.driver.find_element(By.LINK_TEXT, '5') page_5.click() time.sleep(1) rows = self.driver.find_elements(By.XPATH, '//table//tr[td]') self.assertEqual(len(rows), 10) first_item = self.driver.find_element(By.XPATH, '//table//tr[td][1]/td[2]').text last_item = self.driver.find_element(By.XPATH, '//table//tr[td][10]/td[2]').text self.assertEqual(first_item, 'Item 41') self.assertEqual(last_item, 'Item 50') def test_last_page(self): page_10 = self.driver.find_element(By.LINK_TEXT, '10') page_10.click() time.sleep(1) rows = self.driver.find_elements(By.XPATH, '//table//tr[td]') self.assertEqual(len(rows), 10) first_item = self.driver.find_element(By.XPATH, '//table//tr[td][1]/td[2]').text last_item = self.driver.find_element(By.XPATH, '//table//tr[td][10]/td[2]').text self.assertEqual(first_item, 'Item 91') self.assertEqual(last_item, 'Item 100') if __name__ == '__main__': unittest.main()
The test_app.py
module employs Selenium to automate testing of the Flask
application's pagination. The setUp
method configures a test-specific Flask
instance with a temporary database, initializes and populates it, and launches
the server in a separate thread. Selenium's Chrome driver is initialized to
access the app. tearDown
ensures cleanup by closing the browser and removing
the test database.
The test_initial_page_load
verifies the first page loads 10 rows,
with items 'Item 1' to 'Item 10'. The test_pagination_next
confirms
clicking 'Next' shows 'Item 11' to 'Item 20'. The
test_pagination_specific_page
checks page 5 displays 'Item 41' to
'Item 50'. The test_last_page
ensures page 10 shows 'Item 91' to
'Item 100'.
Each test uses XPath to target table rows and asserts the expected content, validating pagination accuracy.
Requirements
Specify the required Python packages in requirements.txt
:
flask sqlite3 selenium webdriver-manager click
Install the dependencies using the following command:
pip install -r requirements.txt
Running the Application
To run the application, set the FLASK_APP
environment variable.
Use set FLASK_APP=flask_pagination
for Windows (Command Prompt),
export FLASK_APP=flask_pagination
for Unix/Linux/macOS, or
$env:FLASK_APP = "flask_pagination"
for Windows PowerShell.
Then, initialize the database with flask init-db
, which outputs
Initialized the database
. Populate the database using flask
populate-db
, and it will confirm Database populated with 100
items
. Start the application with python run.py
and access
it at http://localhost:5000
to view the paginated table.
Run tests using python -m tests.test_app
. The tests generate a
temporary database, perform checks, and clean up afterward.
In this article, we have created a Flask application that uses pagination to display 10 items per page from 100 total rows. Unittests are written for testing this functionality with Selenium library.
Author
List all Python tutorials.