Testing a Submit Form
last modified March 23, 2025
This tutorial shows how to build a Flask app with a user form, store data in SQLite, and test it with Selenium and unittest.
Introduction
Flask is a lightweight Python web framework. Here, we create a simple app to collect user data—first name, last name, occupation, and salary—via a form, save it to a database, and verify functionality with automated tests.
Project Structure
user_form_app/ ├── requirements.txt # Dependencies ├── run.py # App entry point ├── holy_grail_app/ │ ├── config.py # Configuration │ ├── db.py # Database setup │ ├── forms.py # Form definition │ ├── models.py # Database model │ ├── routes.py # Routes │ ├── __init__.py # App factory │ ├── static/ │ │ └── style.css # CSS styling │ └── templates/ │ └── index.html # Form template ├── instance/ │ └── users.db # SQLite database └── tests/ └── test_app.py # Selenium tests
The project is structured to keep the application modular and
organized. The root directory contains requirements.txt
for dependencies and run.py
to launch the app. Inside
holy_grail_app/
, core files like config.py
,
db.py
, and routes.py
handle setup, database,
and routing logic, respectively.
The static/
folder holds style.css
for
styling, while templates/
contains index.html
for the form UI. The instance/
folder stores the SQLite
database (users.db
), and tests/
includes
test_app.py
for automated testing with Selenium.
Flask Application Setup
from flask import Flask import os def create_app(app_config=None): app = Flask(__name__, instance_relative_config=True) try: os.makedirs(app.instance_path) except OSError: pass if app_config is None: env_config = os.getenv("FLASK_ENV", "development") if env_config == "testing": app.config.from_object("holy_grail_app.config.TestingConfig") else: app.config.from_object("holy_grail_app.config.DevelopmentConfig") elif isinstance(app_config, dict): app.config.from_mapping(app_config) else: app.config.from_object(app_config) app.config.from_pyfile("config.py", silent=True) from . import db db.init_app(app) from . import routes app.register_blueprint(routes.bp) return app
This file defines the application factory function create_app
,
which initializes the Flask app. The instance_relative_config=True
parameter tells Flask to look for configuration files in the instance/
folder, where users.db
resides. The os.makedirs
call
ensures this folder exists, silently skipping if it already does.
Configuration is loaded dynamically: if no app_config
is provided,
it checks the FLASK_ENV
environment variable (defaulting to
"development") and selects either DevelopmentConfig
or
TestingConfig
. It can also accept a dictionary or config object.
The database is initialized via db.init_app
, and routes are
registered using a Blueprint from routes.py
.
Configuration
class Config: SECRET_KEY = 'your_default_secret_key' SQLALCHEMY_TRACK_MODIFICATIONS = False class DevelopmentConfig(Config): DEBUG = True DATABASE = 'users.db' SQLALCHEMY_DATABASE_URI = 'sqlite:///users.db' class TestingConfig(Config): TESTING = True DATABASE = 'test_users.db' SQLALCHEMY_DATABASE_URI = 'sqlite:///test_users.db' WTF_CSRF_ENABLED = False
The Config
base class sets a SECRET_KEY
for security
(should be unique in production) and disables SQLALCHEMY_TRACK_MODIFICATIONS
to optimize performance. DevelopmentConfig
inherits these settings,
enables DEBUG
for development features like auto-reloading, and
points to users.db
via SQLALCHEMY_DATABASE_URI
.
TestingConfig
is tailored for testing, setting TESTING
to True
, using a separate test_users.db
to isolate test
data, and disabling WTF_CSRF_ENABLED
to simplify form submission
in tests. These settings are loaded by create_app
based on the
environment.
Database Setup
This module integrates SQLite with Flask-SQLAlchemy.
import sqlite3 import click from flask import current_app, g from flask_sqlalchemy import SQLAlchemy from sqlalchemy import text db = SQLAlchemy() 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_sql = db.get_engine() with db_sql.connect() as conn: conn.execute(text('DROP TABLE IF EXISTS users')) conn.execute(text( 'CREATE TABLE users (id INTEGER PRIMARY KEY, first_name TEXT, ' 'last_name TEXT, occupation TEXT, salary INTEGER)')) conn.commit() def init_app(app): app.teardown_appcontext(close_db) app.cli.add_command(init_db_command) db.init_app(app) @click.command('init-db') def init_db_command(): init_db() click.echo('Initialized the database.')
The db = SQLAlchemy()
line creates an ORM instance used throughout
the app. get_db
manages SQLite connections, storing them in
g
(Flask's request context) to avoid reopening during a request,
and sets row_factory
for dictionary-like row access.
close_db
ensures connections close after each request.
init_db
drops and recreates the users
table with columns
for id
, first_name
, last_name
,
occupation
, and salary
, matching the app's data model.
init_app
ties this setup to the Flask app, adding a teardown function
and a CLI command (flask init-db
) via click
to initialize
the database manually.
User Model
The User
class defines the database model using Flask-SQLAlchemy,
mapping to the users
table created in db.py
.
from .db import db class User(db.Model): __tablename__ = 'users' id = db.Column(db.Integer, primary_key=True) first_name = db.Column(db.String(50), nullable=False) last_name = db.Column(db.String(50), nullable=False) occupation = db.Column(db.String(100), nullable=False) salary = db.Column(db.Integer, nullable=False)
The id
column is an auto-incrementing integer primary key, while
first_name
and last_name
are strings limited to 50
characters, occupation
to 100 characters, and salary
is an integer—all marked nullable=False
to require values.
This model directly corresponds to the form fields and database schema, ensuring
data submitted via the form can be stored and queried consistently. It's used
in routes.py
to create and retrieve user records.
User Form
UserForm
leverages Flask-WTF and WTForms to define the user input
form. Each field—first_name
, last_name
,
occupation
, and salary
—matches the User
model's columns. StringField
handles text inputs, while
IntegerField
ensures salary
is numeric.
SubmitField
creates the submit button.
from flask_wtf import FlaskForm from wtforms import StringField, IntegerField, SubmitField from wtforms.validators import DataRequired, Length class UserForm(FlaskForm): first_name = StringField('First Name', validators=[ DataRequired(), Length(max=50)]) last_name = StringField('Last Name', validators=[ DataRequired(), Length(max=50)]) occupation = StringField('Occupation', validators=[ DataRequired(), Length(max=100)]) salary = IntegerField('Salary', validators=[DataRequired()]) submit = SubmitField('Submit')
Validators like DataRequired()
ensure fields aren't empty, and
Length(max=...)
enforces the same character limits as the model
(50 for names, 100 for occupation). This form is rendered in index.html
and validated in routes.py
before saving data.
HTML Template
This HTML template renders the UserForm
defined in
forms.py
.
<!DOCTYPE html> <html> <head> <title>User Form</title> <link rel="stylesheet" href="{{ url_for('static', filename='style.css') }}"> </head> <body> <div class="form-container"> <form method="POST" class="pure-form"> <h1>User Form</h1> {{ form.hidden_tag() }} <div> {{ form.first_name.label }} {{ form.first_name(size=20) }} {% if form.first_name.errors %} <ul class="errors"> {% for error in form.first_name.errors %} <li>{{ error }}</li> {% endfor %} </ul> {% endif %} </div> <div> {{ form.last_name.label }} {{ form.last_name(size=20) }} {% if form.last_name.errors %} <ul class="errors"> {% for error in form.last_name.errors %} <li>{{ error }}</li> {% endfor %} </ul> {% endif %} </div> <div> {{ form.occupation.label }} {{ form.occupation(size=20) }} {% if form.occupation.errors %} <ul class="errors"> {% for error in form.occupation.errors %} <li>{{ error }}</li> {% endfor %} </ul> {% endif %} </div> <div> {{ form.salary.label }} {{ form.salary(size=20) }} {% if form.salary.errors %} <ul class="errors"> {% for error in form.salary.errors %} <li>{{ error }}</li> {% endfor %} </ul> {% endif %} </div> {{ form.submit() }} </form> </div> </body> </html>
The <link>
tag uses Flask's url_for
to load
style.css
from the static/
folder. The form is wrapped
in a <div class="form-container">
for centering, and the
<form>
tag uses method="POST"
to submit data to
the /
route.
Jinja2 syntax integrates the form fields (form.first_name
, etc.),
displaying labels and inputs with a size=20
attribute for width.
The hidden_tag()
adds a CSRF token for security. Error handling
uses conditionals to show validation errors in a <ul class="errors">
,
styled red by the CSS, ensuring users see feedback on invalid input.
CSS Styling
The CSS styles the form for a clean, user-friendly look.
.errors { color: red; list-style-type: none; padding: 0; } .form-container { display: flex; justify-content: center; align-items: center; height: 100vh; width: 100vw; background-color: #f5f5f5; margin: 0; } .pure-form { background: white; padding: 20px; border-radius: 8px; box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1); width: 300px; } .pure-form label { margin: 0 0 5px 0; display: block; font-weight: bold; } .pure-form input { margin-bottom: 10px; padding: 5px; width: 100%; border: 1px solid #ccc; border-radius: 4px; } .pure-form button { padding: 10px 15px; background-color: #4CAF50; color: white; border: none; border-radius: 4px; cursor: pointer; font-weight: bold; width: 100%; } .pure-form button:hover { background-color: #45a049; }
The .errors
class formats validation messages in red, removing list
bullets for simplicity. .form-container
uses Flexbox to center the
form both horizontally and vertically, filling the viewport (100vh
,
100vw
) with a light gray background.
.pure-form
styles the form itself with a white background, padding,
rounded corners, and a subtle shadow, fixing its width at 300px. Labels are bold
and block-level, inputs span the full width with light borders, and the button
is green with a hover effect, enhancing usability and visual appeal.
Routes
This module uses a Blueprint
named main
to define the
app's routes.
from flask import Blueprint, render_template, redirect, url_for from .models import User from .forms import UserForm from .db import db bp = Blueprint('main', __name__) @bp.route('/', methods=['GET', 'POST']) def index(): form = UserForm() if form.validate_on_submit(): user = User( first_name=form.first_name.data, last_name=form.last_name.data, occupation=form.occupation.data, salary=form.salary.data ) db.session.add(user) db.session.commit() return redirect(url_for('main.success', user_id=user.id)) return render_template('index.html', form=form) @bp.route('/success/<int:user_id>') def success(user_id): user = User.query.get_or_404(user_id) return f'User {user.first_name} {user.last_name} added successfully!'
The /
route handles both GET
(displaying
the form) and POST
(submitting it). On GET
, it creates
a UserForm
instance and renders index.html
. On a valid
POST
, it constructs a User
object from form data, adds
it to the database, and redirects to the success
route.
The /success/<int:user_id>
route takes a user ID, retrieves
the corresponding User
record (or returns 404 if not found), and
displays a simple success message. This logic ties the form, model, and database
together, completing the app's workflow.
Selenium Tests
This test file uses unittest
and Selenium to verify the app's
functionality.
# tests/test_app.py 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 from selenium.webdriver.support.ui import WebDriverWait from selenium.webdriver.support import expected_conditions as EC import time import threading import os from holy_grail_app.db import db from holy_grail_app.models import User from holy_grail_app import create_app from holy_grail_app.config import TestingConfig class TestUserForm(unittest.TestCase): def setUp(self): self.app = create_app(TestingConfig) self.client = self.app.test_client() # Initialize database with self.app.app_context(): from holy_grail_app.db import init_db init_db() # Start Flask server in a thread self.server_thread = threading.Thread( target=self.app.run, kwargs={'port': 5000}) self.server_thread.daemon = True self.server_thread.start() time.sleep(1) # Wait for server 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() with self.app.app_context(): # Get the absolute path to the database file db_path = os.path.abspath( os.path.join(self.app.instance_path, self.app.config['DATABASE']) ) print(f"Absolute database path: {db_path}") # Debugging output try: db.engine.dispose() # Dispose of database connections if os.path.exists(db_path): os.remove(db_path) print(f"Database file '{db_path}' removed successfully.") else: print(f"Database file '{db_path}' does not exist.") except Exception as e: print(f"Error during cleanup: {e}") def test_index_get(self): """Test the index page loads with all form fields.""" driver = self.driver # Wait for form to load WebDriverWait(driver, 10).until( EC.presence_of_element_located((By.CLASS_NAME, "pure-form")) ) # Check form elements self.assertIn("User Form", driver.page_source) self.assertTrue(driver.find_element(By.ID, "first_name")) self.assertTrue(driver.find_element(By.ID, "last_name")) self.assertTrue(driver.find_element(By.ID, "occupation")) self.assertTrue(driver.find_element(By.ID, "salary")) self.assertTrue(driver.find_element(By.ID, "submit")) def test_user_model(self): """Test creating and querying a User in the database.""" with self.app.app_context(): user = User(first_name="Alice", last_name="Johnson", occupation="Designer", salary=75000.0) db.session.add(user) db.session.commit() queried_user = User.query.filter_by(first_name="Alice").first() self.assertIsNotNone(queried_user) self.assertEqual(queried_user.last_name, "Johnson") self.assertEqual(queried_user.occupation, "Designer") self.assertEqual(queried_user.salary, 75000.0) def test_form_submission_valid(self): """Test submitting valid form data redirects to success.""" driver = self.driver # driver.get('http://localhost:5001/') # Wait for form to be interactive WebDriverWait(driver, 10).until( EC.element_to_be_clickable((By.ID, "submit")) ) # Fill form driver.find_element(By.ID, "first_name").send_keys("Jane") driver.find_element(By.ID, "last_name").send_keys("Smith") driver.find_element(By.ID, "occupation").send_keys("Developer") driver.find_element(By.ID, "salary").send_keys("60000") # Submit form driver.find_element(By.ID, "submit").click() # Wait for redirect WebDriverWait(driver, 10).until( EC.text_to_be_present_in_element((By.TAG_NAME, "body"), "Jane Smith") ) # Verify success page self.assertIn("User Jane Smith added successfully!", driver.page_source) # Verify database with self.app.app_context(): user = User.query.filter_by(first_name="Jane").first() self.assertIsNotNone(user) self.assertEqual(user.last_name, "Smith") self.assertEqual(user.occupation, "Developer") self.assertEqual(user.salary, 60000.0) def test_form_submission_missing_field(self): """Test submitting with a missing field shows validation error.""" driver = self.driver # Wait for form to be interactive WebDriverWait(driver, 10).until( EC.presence_of_element_located((By.CLASS_NAME, "pure-form")) ) # Remove 'required' attributes to bypass client-side validation driver.execute_script(""" document.querySelectorAll('input[required]').forEach(input => { input.removeAttribute('required'); }); """) # Fill partial form (missing first_name) driver.find_element(By.ID, "last_name").send_keys("Smith") driver.find_element(By.ID, "occupation").send_keys("Developer") driver.find_element(By.ID, "salary").send_keys("60000") # Submit form driver.find_element(By.ID, "submit").click() # Wait for error message WebDriverWait(driver, 10).until( EC.presence_of_element_located( (By.XPATH, "//ul[@class='errors']/li")) ) self.assertIn("This field is required", driver.page_source) self.assertIn("User Form", driver.page_source) def test_form_submission_invalid_salary(self): """Test submitting a non-numeric salary shows an error.""" driver = self.driver # Wait for form to be interactive WebDriverWait(driver, 10).until( EC.presence_of_element_located((By.CLASS_NAME, "pure-form")) ) # Remove 'required' attributes to bypass client-side validation driver.execute_script(""" document.querySelectorAll('input[required]').forEach(input => { input.removeAttribute('required'); }); """) # Fill form with invalid salary driver.find_element(By.ID, "first_name").send_keys("Jane") driver.find_element(By.ID, "last_name").send_keys("Smith") driver.find_element(By.ID, "occupation").send_keys("Developer") driver.find_element(By.ID, "salary").send_keys("invalid") # Submit form driver.find_element(By.ID, "submit").click() # Wait for error message WebDriverWait(driver, 10).until( EC.presence_of_element_located( (By.XPATH, "//ul[@class='errors']/li")) ) self.assertIn("This field is required.", driver.page_source) self.assertIn("User Form", driver.page_source) def test_form_submission_length_exceeded(self): """Test submitting a too-long field shows an error.""" driver = self.driver # Wait for form to be interactive WebDriverWait(driver, 10).until( EC.presence_of_element_located((By.CLASS_NAME, "pure-form")) ) # Remove 'required' attributes to bypass client-side validation driver.execute_script(""" document.querySelectorAll('input[required]').forEach(input => { input.removeAttribute('required'); }); """) # Fill form with oversized first_name driver.execute_script( "document.getElementById('first_name').value = 'a'.repeat(51);" ) driver.find_element(By.ID, "last_name").send_keys("Smith") driver.find_element(By.ID, "occupation").send_keys("Developer") driver.find_element(By.ID, "salary").send_keys("60000") # Submit form driver.find_element(By.ID, "submit").click() # Wait for error message WebDriverWait(driver, 10).until( EC.presence_of_element_located( (By.XPATH, "//ul[@class='errors']/li")) ) self.assertIn("Field cannot be longer than 50 characters", driver.page_source) self.assertIn("User Form", driver.page_source) if __name__ == '__main__': unittest.main()
The setUp
method initializes the test environment by creating
the Flask application with the testing configuration, initializing the database,
starting the Flask server in a separate thread, and launching a Chrome WebDriver
instance to interact with the application. This ensures a fresh and functional
environment for each test.
The tearDown
method is responsible for cleaning up the test
environment after each test. It quits the WebDriver, closes active database
connections, disposes of the database engine, and attempts to delete the test
database file to maintain isolation between tests and avoid lingering
resources.
The test_index_get
method verifies that the index page loads
correctly and contains all the necessary form fields. It checks for the presence
of the form and ensures all expected elements, such as input fields and the
submit button, are available and accessible in the page source.
The test_user_model
method tests the functionality of the
User
model by adding a user to the database and then querying it.
It confirms that the user is successfully added and that all the attributes
match the expected values, validating the model's behavior.
The test_form_submission_valid
method tests the complete
workflow of submitting a valid form. It fills in all the required fields with
valid data, submits the form, and verifies that the user is redirected to a
success page. It also ensures the submitted data is correctly stored in the
database.
The test_form_submission_missing_field
method evaluates the
form's validation mechanism when a required field is missing. It submits a form
without filling out the first_name
field and verifies that an
appropriate validation error message is displayed on the page.
The test_form_submission_invalid_salary
method checks how the
form handles invalid data in the salary field. It submits the form with a
non-numeric salary value and validates that the application displays an
appropriate error message to guide the user.
The test_form_submission_length_exceeded
method ensures the form
properly validates field lengths. It submits a form with an excessively long
value for the first_name
field and verifies that the error message
indicating the maximum length constraint is correctly displayed.
Running the App and Tests
$ pip install -r requirements.txt $ flask init-db $ python run.py
To run tests:
$ python -m unittest tests/test_app.py -v
First, install dependencies listed in requirements.txt
using pip.
Then, run flask init-db
to set up the users.db
database.
Finally, launch the app with python run.py
to start the development
server. For testing, use unittest
with the -v
flag for
verbose output, ensuring Selenium tests run against the app.
Author
List all Python tutorials.