Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/mjhea0/flaskr-tdd

Flaskr: Intro to Flask, Test-Driven Development (TDD), and JavaScript
https://github.com/mjhea0/flaskr-tdd

flask javascript python tdd test-driven-development

Last synced: 3 days ago
JSON representation

Flaskr: Intro to Flask, Test-Driven Development (TDD), and JavaScript

Awesome Lists containing this project

README

        

# Flaskr - Intro to Flask, Test-Driven Development, and JavaScript

As many of you know, Flaskr -- a mini-blog-like-app -- is the app that you build for the official Flask [tutorial](https://flask.palletsprojects.com/en/3.0.x/tutorial/). I've gone through the tutorial more times than I care to admit. Anyway, I wanted to take the tutorial a step further by adding [Test-Driven Development](https://testdriven.io/test-driven-development/) (TDD), a bit of JavaScript, and deployment. This article is that tutorial. Enjoy.

Also, if you're completely new to Flask and/or web development in general, it's important to grasp these basic fundamental concepts:

1. The difference between HTTP GET and POST requests and how functions within the app handle each.
1. What HTTP "requests" and "responses" are.
1. How HTML pages are rendered and/or returned to the end user.

> This project is powered by **[TestDriven.io](https://testdriven.io/)**. Please support this open source project by purchasing one of our Flask courses. Learn how to build, test, and deploy microservices powered by Docker, Flask, and React!

## What you're building

You'll be building a simple blogging app in this tutorial:

![flaskr app](/flaskr-app.png)

## Changelog

This tutorial was last updated on October 17th, 2023:

- **10/17/2023**:
- Updated to Python 3.12.0 and bumped all other dependencies.
- **06/03/2022**:
- Updated to Python 3.10.4 and bumped all other dependencies.
- **10/14/2020**:
- Renamed *app.test.py* to *app_test.py*. (Fixed issue #[58](https://github.com/mjhea0/flaskr-tdd/issues/58).)
- Updated to Python 3.9 and bumped all other dependencies.
- Added pytest v7.1.2. (Fixed issue #[60](https://github.com/mjhea0/flaskr-tdd/issues/60))
- Migrated from `os.path` to `pathlib`.
- **11/05/2019**:
- Updated to Python 3.8.0, Flask 1.1.1, and Bootstrap 4.3.1.
- Replaced jQuery with vanilla JavaScript.
- Added Black and Flake8.
- Used Postgres in production.
- Restricted post delete requests.
- **10/07/2018**: Updated to Python 3.7.0.
- **05/10/2018**: Updated to Python 3.6.5, Flask 1.0.2, Bootstrap 4.1.1.
- **10/16/2017**:
- Updated to Python 3.6.2.
- Updated to Bootstrap 4.
- **10/10/2017**: Added a search feature.
- **07/03/2017**: Updated to Python 3.6.1.
- **01/24/2016**: Updated to Python 3 (v3.5.1)!
- **08/24/2014**: PEP8 updates.
- **02/25/2014**: Upgraded to SQLAlchemy.
- **02/20/2014**: Completed AJAX.
- **12/06/2013**: Added Bootstrap 3 styles
- **11/29/2013**: Updated unit tests.
- **11/19/2013**: Fixed typo. Updated unit tests.
- **11/11/2013**: Added information on requests.

## Contents

1. [Test Driven Development?](#test-driven-development)
1. [Download Python](#download-python)
1. [Project Setup](#project-setup)
1. [First Test](#first-test)
1. [Flaskr Setup](#flaskr-setup)
1. [Second Test](#second-test)
1. [Database Setup](#database-setup)
1. [Templates and Views](#templates-and-views)
1. [Add Some Style](#add-some-style)
1. [JavaScript](#javascript)
1. [Deployment](#deployment)
1. [Bootstrap](#bootstrap)
1. [SQLAlchemy](#sqlalchemy)
1. [Search Page](#search-page)
1. [Login Required](#login-required)
1. [Postgres Heroku](#postgres-heroku)
1. [Linting and Code Formatting](#linting-and-code-formatting)
1. [Conclusion](#conclusion)

## Requirements

This tutorial utilizes the following requirements:

1. Python v3.12.0
1. Flask v3.0.0
1. Flask-SQLAlchemy v3.1.1
1. Gunicorn v21.2.0
1. Psycopg2 v2.9.9
1. Flake8 v6.1.0
1. Black v23.10.0
1. pytest v7.4.2

## Test Driven Development?

![tdd](https://raw.githubusercontent.com/mjhea0/flaskr-tdd/master/tdd.png)

Test-Driven Development (TDD) is an iterative development cycle that emphasizes writing automated tests before writing the actual feature or function. Put another way, TDD combines building and testing. This process not only helps ensure correctness of the code -- but also helps to indirectly evolve the design and architecture of the project at hand.

TDD usually follows the "Red-Green-Refactor" cycle, as shown in the image above:

1. Write a test
1. Run the test (it should fail)
1. Write just enough code for the test to pass
1. Refactor code and retest, again and again (if necessary)

> For more, check out [What is Test-Driven Development?](https://testdriven.io/test-driven-development/).

## Download Python

Before beginning make sure you have the latest version of [Python 3.12](https://www.python.org/downloads/release/python-3120/) installed, which you can download from [http://www.python.org/download/](http://www.python.org/download/).

> This tutorial uses Python v3.12.0.

Along with Python, the following tools are also installed:

- [pip](https://pip.pypa.io/en/stable/) - a [package management](http://en.wikipedia.org/wiki/Package_management_system) system for Python, similar to gem or npm for Ruby and Node, respectively.
- [venv](https://docs.python.org/3/library/venv.html) - used to create isolated environments for development. This is standard practice. Always, always, ALWAYS utilize virtual environments. If you don't, you'll eventually run into problems with dependency conflicts.

> Feel free to swap out virtualenv and Pip for [Poetry](https://python-poetry.org) or [Pipenv](https://github.com/pypa/pipenv). For more, review [Modern Python Environments](https://testdriven.io/blog/python-environments/).

## Project Setup

Create a new directory to store the project:

```sh
$ mkdir flaskr-tdd
$ cd flaskr-tdd
```

Create and activate a virtual environment:

```sh
$ python3.12 -m venv env
$ source env/bin/activate
(env)$
```

> You know that you're in a virtual environment when `env` is displayed before the `$` in your terminal: `(env)$`. To exit the virtual environment, use the command `deactivate`. You can reactivate by navigating back to the project directory and running `source env/bin/activate`.

Install Flask with pip:

```sh
(env)$ pip install flask==3.0.0
```

## First Test

Let's start with a simple "hello, world" app.

Create the following files and folders:

```sh
├── project
│   ├── __init__.py
│   ├── app.py
└── tests
├── __init__.py
└── app_test.py
```

While the Python standard library comes with a unit testing framework called ùnittest, [pytest](https://pytest.org/) is the go-to testing framework for testing Python code.

> For more on pytest, check out [Pytest for Beginners](https://testdriven.io/blog/pytest-for-beginners/).

Install it:

```sh
(env)$ pip install pytest==7.4.2
```

Open *tests/app_test.py* in your favorite text editor -- like [Visual Studio Code](https://code.visualstudio.com/), [Sublime Text](https://www.sublimetext.com/), or [PyCharm](https://www.jetbrains.com/pycharm/) -- and then add the following code:

```python
from project.app import app

def test_index():
tester = app.test_client()
response = tester.get("/", content_type="html/text")

assert response.status_code == 200
assert response.data == b"Hello, World!"
```

Essentially, we're testing whether the response that we get back has a status code of "200" and that "Hello, World!" is displayed.

Run the test:

```sh
(env)$ python -m pytest
```

If all goes well, this test will fail:

```sh
ImportError: cannot import name 'app' from 'project.app
```

Now add the code for this to pass to *project/app.py*:

```python
from flask import Flask

# create and initialize a new Flask app
app = Flask(__name__)

@app.route("/")
def hello():
return "Hello, World!"

if __name__ == "__main__":
app.run()
```

Run the app:

```sh
(env)$ FLASK_APP=project/app.py python -m flask run -p 5001
```

> The `FLASK_APP` environment variable is used to tell Flask to look for the application in a different module.

Then, navigate to [http://localhost:5001/](http://localhost:5001/) in your browser of choice. You should see "Hello, World!" on your screen.

Return to the terminal. Kill the server with Ctrl+C.

Run the test again:

```sh
(env)$ python -m pytest

=============================== test session starts ===============================
platform darwin -- Python 3.10.4, pytest-7.4.2, pluggy-1.0.0
rootdir: /Users/michael/repos/github/flaskr-tdd
collected 1 item

tests/app_test.py . [100%]

================================ 1 passed in 0.10s ================================
```

Nice.

## Database Setup

Essentially, we want to open a database connection, create the database based on a defined schema if it doesn't already exist, and then close the connection each time a test is ran.

Create a new file called *schema.sql* in "project" and add the following code:

```sql
drop table if exists entries;

create table entries (
id integer primary key autoincrement,
title text not null,
text text not null
);
```

This will set up a single table with three fields: "id", "title", and "text". SQLite will be used for our RDMS since it's part of the standard Python library and requires no configuration.

Update *app.py*:

```python
from flask import Flask

# configuration
DATABASE = "flaskr.db"

# create and initialize a new Flask app
app = Flask(__name__)

# load the config
app.config.from_object(__name__)

@app.route("/")
def hello():
return "Hello, World!"

if __name__ == "__main__":
app.run()
```

Here, we created a configuration section for config variables (with the name of the future SQLite database) and loaded the config after app initialization.

How do we test for the existence of a file? Update *app_test.py* like so:

```python
from pathlib import Path

from project.app import app

def test_index():
tester = app.test_client()
response = tester.get("/", content_type="html/text")

assert response.status_code == 200
assert response.data == b"Hello, World!"

def test_database():
assert Path("flaskr.db").is_file()
```

Run it to make sure it fails, indicating that the database does not exist.

Now add the following code to *app.py*, just before the `hello` view function:

```python
# connect to database
def connect_db():
"""Connects to the database."""
rv = sqlite3.connect(app.config["DATABASE"])
rv.row_factory = sqlite3.Row
return rv

# create the database
def init_db():
with app.app_context():
db = get_db()
with app.open_resource("schema.sql", mode="r") as f:
db.cursor().executescript(f.read())
db.commit()

# open database connection
def get_db():
if not hasattr(g, "sqlite_db"):
g.sqlite_db = connect_db()
return g.sqlite_db

# close database connection
@app.teardown_appcontext
def close_db(error):
if hasattr(g, "sqlite_db"):
g.sqlite_db.close()
```

Add the imports:

```python
import sqlite3

from flask import Flask, g
```

> Curious about `g` object? Check out the [Understanding the Application and Request Contexts in Flask](https://testdriven.io/blog/flask-contexts/) for more.

You should now have:

```python
import sqlite3

from flask import Flask, g

# configuration
DATABASE = "flaskr.db"

# create and initialize a new Flask app
app = Flask(__name__)

# load the config
app.config.from_object(__name__)

# connect to database
def connect_db():
"""Connects to the database."""
rv = sqlite3.connect(app.config["DATABASE"])
rv.row_factory = sqlite3.Row
return rv

# create the database
def init_db():
with app.app_context():
db = get_db()
with app.open_resource("schema.sql", mode="r") as f:
db.cursor().executescript(f.read())
db.commit()

# open database connection
def get_db():
if not hasattr(g, "sqlite_db"):
g.sqlite_db = connect_db()
return g.sqlite_db

# close database connection
@app.teardown_appcontext
def close_db(error):
if hasattr(g, "sqlite_db"):
g.sqlite_db.close()

@app.route("/")
def hello():
return "Hello, World!"

if __name__ == "__main__":
app.run()
```

Now, create a database by starting up a Python shell and importing and then calling the `init_db` function:

```python
>>> from project.app import init_db
>>> init_db()
```

Close the shell, then run the test again. Does it pass? It should. Now we know that the database has been created.

You can also call `init_db` within the test, to ensure that the test can be ran independently:

```python
from pathlib import Path

from project.app import app, init_db

def test_index():
tester = app.test_client()
response = tester.get("/", content_type="html/text")

assert response.status_code == 200
assert response.data == b"Hello, World!"

def test_database():
init_db()
assert Path("flaskr.db").is_file()
```

Updated structure:

```sh
├── flaskr.db
├── project
│   ├── __init__.py
│   ├── app.py
│   └── schema.sql
└── tests
├── __init__.py
└── app_test.py
```

## Templates and Views

Next, we need to set up the templates and the associated views, which define the routes. Think about this from a user's standpoint:

1. Users should be able to log in and out.
1. Once logged in, users should be able to post new messages.
1. Finally, users should be able to view the messages.

Write some tests for this first.

### Tests

Take a look at the final code below. I added docstrings for explanation.

```python
import os
import pytest
from pathlib import Path

from project.app import app, init_db

TEST_DB = "test.db"

@pytest.fixture
def client():
BASE_DIR = Path(__file__).resolve().parent.parent
app.config["TESTING"] = True
app.config["DATABASE"] = BASE_DIR.joinpath(TEST_DB)

init_db() # setup
yield app.test_client() # tests run here
init_db() # teardown

def login(client, username, password):
"""Login helper function"""
return client.post(
"/login",
data=dict(username=username, password=password),
follow_redirects=True,
)

def logout(client):
"""Logout helper function"""
return client.get("/logout", follow_redirects=True)

def test_index(client):
response = client.get("/", content_type="html/text")
assert response.status_code == 200

def test_database(client):
"""initial test. ensure that the database exists"""
tester = Path("test.db").is_file()
assert tester

def test_empty_db(client):
"""Ensure database is blank"""
rv = client.get("/")
assert b"No entries yet. Add some!" in rv.data

def test_login_logout(client):
"""Test login and logout using helper functions"""
rv = login(client, app.config["USERNAME"], app.config["PASSWORD"])
assert b"You were logged in" in rv.data
rv = logout(client)
assert b"You were logged out" in rv.data
rv = login(client, app.config["USERNAME"] + "x", app.config["PASSWORD"])
assert b"Invalid username" in rv.data
rv = login(client, app.config["USERNAME"], app.config["PASSWORD"] + "x")
assert b"Invalid password" in rv.data

def test_messages(client):
"""Ensure that user can post messages"""
login(client, app.config["USERNAME"], app.config["PASSWORD"])
rv = client.post(
"/add",
data=dict(title="", text="HTML allowed here"),
follow_redirects=True,
)
assert b"No entries here so far" not in rv.data
assert b"<Hello>" in rv.data
assert b"HTML allowed here" in rv.data
```

Take note of the `client` function. This is a pytest [fixture](https://docs.pytest.org/en/stable/fixture.html), which sets up a known state for each test function before the test runs.

Run the tests now:

```sh
(env)$ python -m pytest
```

Three tests should fail:

```sh
=============================== test session starts ===============================
platform darwin -- Python 3.10.4, pytest-7.4.2, pluggy-1.0.0
rootdir: /Users/michael/repos/github/flaskr-tdd
collected 5 items

tests/app_test.py ..FFF [100%]

==================================== FAILURES =====================================
__________________________________ test_empty_db __________________________________

client = >

def test_empty_db(client):
"""Ensure database is blank"""
rv = client.get("/")
> assert b"No entries yet. Add some!" in rv.data
E AssertionError: assert b'No entries yet. Add some!' in b'Hello, World!'
E + where b'Hello, World!' = .data

tests/app_test.py:49: AssertionError
________________________________ test_login_logout ________________________________

client = >

def test_login_logout(client):
"""Test login and logout using helper functions"""
> rv = login(client, app.config["USERNAME"], app.config["PASSWORD"])
E KeyError: 'USERNAME'

tests/app_test.py:54: KeyError
__________________________________ test_messages __________________________________

client = >

def test_messages(client):
"""Ensure that user can post messages"""
> login(client, app.config["USERNAME"], app.config["PASSWORD"])
E KeyError: 'USERNAME'

tests/app_test.py:66: KeyError
============================= short test summary info =============================
FAILED tests/app_test.py::test_empty_db -
AssertionError: assert b'No entries yet. Add some!' in b'Hello, World!'
FAILED tests/app_test.py::test_login_logout - KeyError: 'USERNAME'
FAILED tests/app_test.py::test_messages - KeyError: 'USERNAME'
=========================== 3 failed, 2 passed in 0.17s ==========================
```

Let's get these all green, one at a time...

### Show Entries

First, replace the `hello` view with the following view function for displaying the entries to *app.py*:

```python
@app.route('/')
def index():
"""Searches the database for entries, then displays them."""
db = get_db()
cur = db.execute('select * from entries order by id desc')
entries = cur.fetchall()
return render_template('index.html', entries=entries)
```

Import in `render_template`:

```python
from flask import Flask, g, render_template
```

Then, create a new folder called "templates" inside of "project", and add an *index.html* template file to it:

```html


Flaskr




Flaskr-TDD


{% if not session.logged_in %}
log in
{% else %}
log out
{% endif %}

{% for message in get_flashed_messages() %}

{{ message }}

{% endfor %} {% block body %}{% endblock %} {% if session.logged_in %}


Title:


Text:





{% endif %}


    {% for entry in entries %}

  • {{ entry.title }}


    {{ entry.text|safe }}

  • {% else %}
  • No entries yet. Add some!

  • {% endfor %}


```

### User Login and Logout

Update *app.py*:

```python
@app.route('/login', methods=['GET', 'POST'])
def login():
"""User login/authentication/session management."""
error = None
if request.method == 'POST':
if request.form['username'] != app.config['USERNAME']:
error = 'Invalid username'
elif request.form['password'] != app.config['PASSWORD']:
error = 'Invalid password'
else:
session['logged_in'] = True
flash('You were logged in')
return redirect(url_for('index'))
return render_template('login.html', error=error)

@app.route('/logout')
def logout():
"""User logout/authentication/session management."""
session.pop('logged_in', None)
flash('You were logged out')
return redirect(url_for('index'))
```

In the above `login` function, the decorator indicates that the route can accept either a GET or POST request. Put simply, a request is initiated by the end user when they access the `/login` URL. The difference between these requests is simple: GET is used for accessing a webpage, while POST is used when information is sent to the server. Thus, when a user accesses the `/login` URL, they are using a GET request, but when they attempt to log in, a POST request is used.

Update the config as well:

```python
# configuration
DATABASE = "flaskr.db"
USERNAME = "admin"
PASSWORD = "admin"
SECRET_KEY = "change_me"
```

Add the appropriate imports:

```python
from flask import Flask, g, render_template, request, session, flash, redirect, url_for
```

Add the *login.html* template:

```html


Flaskr-TDD | Login




Flaskr


{% if not session.logged_in %}
log in
{% else %}
log out
{% endif %}

{% for message in get_flashed_messages() %}

{{ message }}

{% endfor %} {% block body %}{% endblock %}

Login

{% if error %}

Error: {{ error }}


{% endif %}



Username:


Password:






```

Run the tests again. You should see two errors:

```sh
E werkzeug.routing.BuildError: Could not build url for endpoint 'add_entry'. Did you mean 'login' instead?
```

Next, add in a view for adding entries:

```python
@app.route('/add', methods=['POST'])
def add_entry():
"""Add new post to database."""
if not session.get('logged_in'):
abort(401)
db = get_db()
db.execute(
'insert into entries (title, text) values (?, ?)',
[request.form['title'], request.form['text']]
)
db.commit()
flash('New entry was successfully posted')
return redirect(url_for('index'))
```

Add the appropriate imports:

```python
from flask import Flask, g, render_template, request, session, flash, redirect, url_for, abort
```

Retest:

```sh
=============================== test session starts ===============================
platform darwin -- Python 3.10.4, pytest-7.4.2, pluggy-1.0.0
rootdir: /Users/michael/repos/github/flaskr-tdd
collected 5 items

tests/app_test.py ..... [100%]

================================ 5 passed in 0.16s ================================
```

Perfect.

## Add Some Style

Save the following styles to a new file called *style.css* in a new folder called "project/static":

```css
body {
font-family: sans-serif;
background: #eee;
}

a,
h1,
h2 {
color: #377ba8;
}

h1,
h2 {
font-family: "Georgia", serif;
margin: 0;
}

h1 {
border-bottom: 2px solid #eee;
}

h2 {
font-size: 1.2em;
}

.page {
margin: 2em auto;
width: 35em;
border: 5px solid #ccc;
padding: 0.8em;
background: white;
}

.entries {
list-style: none;
margin: 0;
padding: 0;
}

.entries li {
margin: 0.8em 1.2em;
}

.entries li h2 {
margin-left: -1em;
}

.add-entry {
font-size: 0.9em;
border-bottom: 1px solid #ccc;
}

.add-entry dl {
font-weight: bold;
}

.metanav {
text-align: right;
font-size: 0.8em;
padding: 0.3em;
margin-bottom: 1em;
background: #fafafa;
}

.flash {
background: #cee5f5;
padding: 0.5em;
border: 1px solid #aacbe2;
}

.error {
background: #f0d6d6;
padding: 0.5em;
}
```

Run your app, log in (username/password = "admin"), add a post, log out.

## JavaScript

Next, let's add some JavaScript to make the app slightly more interactive.

Open *index.html* and update the first `

  • ` like so:

    ```html


  • {{ entry.title }}


    {{ entry.text|safe }}

  • ```

    Now, we can use JavaScript to target each `

  • `. First, we need to add the following script to the document just before the closing body tag:

    ```html

    ```

    Create a *main.js* file in your "static" directory and add the following code:

    ```javascript
    (function () {
    console.log("ready!"); // sanity check
    })();

    const postElements = document.getElementsByClassName("entry");

    for (var i = 0; i < postElements.length; i++) {
    postElements[i].addEventListener("click", function () {
    const postId = this.getElementsByTagName("h2")[0].getAttribute("id");
    const node = this;
    fetch(`/delete/${postId}`)
    .then(function (resp) {
    return resp.json();
    })
    .then(function (result) {
    if (result.status === 1) {
    node.parentNode.removeChild(node);
    console.log(result);
    }
    location.reload();
    })
    .catch(function (err) {
    console.log(err);
    });
    });
    }
    ```

    Add a new function in *app.py* to remove the post from the database:

    ```python
    @app.route('/delete/', methods=['GET'])
    def delete_entry(post_id):
    """Delete post from database"""
    result = {'status': 0, 'message': 'Error'}
    try:
    db = get_db()
    db.execute('delete from entries where id=' + post_id)
    db.commit()
    result = {'status': 1, 'message': "Post Deleted"}
    except Exception as e:
    result = {'status': 0, 'message': repr(e)}
    return jsonify(result)
    ```

    Update the imports:

    ```python
    from flask import Flask, g, render_template, request, session, flash, redirect, url_for, abort, jsonify
    ```

    Finally, add a new test:

    ```python
    def test_delete_message(client):
    """Ensure the messages are being deleted"""
    rv = client.get('/delete/1')
    data = json.loads(rv.data)
    assert data["status"] == 1
    ```

    Make sure to add the following import as well: `import json`.

    Manually test this out by running the server and adding two new entries. Click on one of them. It should be removed from the DOM as well as the database. Double check this.

    Then run your automated test suite. It should pass:

    ```sh
    (env)$ python -m pytest

    =============================== test session starts ===============================
    platform darwin -- Python 3.10.4, pytest-7.4.2, pluggy-1.0.0
    rootdir: /Users/michael/repos/github/flaskr-tdd
    collected 6 items

    tests/app_test.py ...... [100%]

    ================================ 6 passed in 0.17s ================================
    ```

    ## Deployment

    With the app in a working state, let's shift gears and deploy the app to [Heroku](https://www.heroku.com). To do this, first [sign up](https://signup.heroku.com/), and then install the [Heroku CLI](https://devcenter.heroku.com/articles/heroku-cli).

    Next, install a production-grade WSGI web server called [Gunicorn](http://gunicorn.org/):

    ```sh
    (env)$ pip install gunicorn==21.2.0
    ```

    Create a [Procfile](https://devcenter.heroku.com/articles/procfile) in the project root:

    ```sh
    (env)$ touch Procfile
    ```

    And add the following code:

    ```sh
    web: gunicorn project.app:app
    ```

    Create a *requirements.txt* file to specify the external dependencies that need to be installed for the app to work:

    ```sh
    (env)$ touch requirements.txt
    ```

    Add the requirements:

    ```
    Flask==3.0.0
    gunicorn==21.2.0
    pytest==7.4.2
    ```

    Create a *.gitignore* file in the project root:

    ```sh
    (env)$ touch .gitignore
    ```

    And include the following files and folders (so they are not included in version control):

    ```sh
    env
    *.pyc
    *.DS_Store
    __pycache__
    test.db
    ```

    To specify the correct Python runtime, add a new file to the project root called *runtime.txt*:

    ```
    python-3.12.0
    ```

    Add a local Git repo:

    ```sh
    (env)$ git init
    (env)$ git add -A
    (env)$ git commit -m "initial"
    ```

    Deploy to Heroku:

    ```sh
    (env)$ heroku create
    (env)$ git push heroku main
    ```

    Let's test this in the cloud. Run `heroku open` to open the app in your default web browser.

    ## Bootstrap

    Let's update the styles with [Bootstrap](http://getbootstrap.com/).

    First, remove the *style.css* stylesheet from both *index.html* and *login.html*. Then add this stylesheet to both files:

    ```html

    ```

    Now, we have full access to all of the Bootstrap helper classes.

    Replace the code in *login.html* with:

    ```html


    Flaskr-TDD | Login







    Flaskr




    {% for message in get_flashed_messages() %}


    {{ message }}

    {% endfor %}

    Login

    {% if error %}


    Error: {{ error }}


    {% endif %}



    Username:




    Password:










    Use "admin" for username and password




    ```

    And replace the code in *index.html* with:

    ```html


    Flaskr







    Flaskr




    {% if not session.logged_in %}
    log in
    {% else %}
    log out
    {% endif %}



    {% for message in get_flashed_messages() %}


    {{ message }}

    {% endfor %} {% if session.logged_in %}


    Title:




    Text:












    {% endif %}



      {% for entry in entries %}

    • {{ entry.title }}


      {{ entry.text|safe }}

    • {% else %}
    • No entries yet. Add some!

    • {% endfor %}



    ```

    Run the app locally:

    ```sh
    (env)$ FLASK_APP=project/app.py python -m flask run -p 5001
    ```

    Check out the changes in the browser!

    ## SQLAlchemy

    Let's upgrade to [Flask-SQLAlchemy](https://flask-sqlalchemy.palletsprojects.com/), in order to better manage the database.

    ### Setup

    Start by installing Flask-SQLAlchemy:

    ```sh
    (env)$ pip install Flask-SQLAlchemy==3.1.1
    ```

    Make sure to add it to your requirements file as well.

    Next, add a *create_db.py* file to the project root. Then, add the following code:

    ```python
    # create_db.py

    from project.app import app, db
    from project.models import Post

    with app.app_context():
    # create the database and the db table
    db.create_all()

    # commit the changes
    db.session.commit()
    ```

    This file will be used to create our new database. Go ahead and delete the old database file (*flaskr.db*) along with the *project/schema.sql* file.

    Next, add a *project/models.py* file, which will be used to generate the new schema:

    ```python
    from project.app import db

    class Post(db.Model):
    id = db.Column(db.Integer, primary_key=True)
    title = db.Column(db.String, nullable=False)
    text = db.Column(db.String, nullable=False)

    def __init__(self, title, text):
    self.title = title
    self.text = text

    def __repr__(self):
    return f''
    ```

    ### Update *app.py*

    ```python
    import sqlite3
    from pathlib import Path

    from flask import Flask, g, render_template, request, session, \
    flash, redirect, url_for, abort, jsonify
    from flask_sqlalchemy import SQLAlchemy

    basedir = Path(__file__).resolve().parent

    # configuration
    DATABASE = "flaskr.db"
    USERNAME = "admin"
    PASSWORD = "admin"
    SECRET_KEY = "change_me"
    SQLALCHEMY_DATABASE_URI = f'sqlite:///{Path(basedir).joinpath(DATABASE)}'
    SQLALCHEMY_TRACK_MODIFICATIONS = False

    # create and initialize a new Flask app
    app = Flask(__name__)
    # load the config
    app.config.from_object(__name__)
    # init sqlalchemy
    db = SQLAlchemy(app)

    from project import models

    @app.route('/')
    def index():
    """Searches the database for entries, then displays them."""
    entries = db.session.query(models.Post)
    return render_template('index.html', entries=entries)

    @app.route('/add', methods=['POST'])
    def add_entry():
    """Adds new post to the database."""
    if not session.get('logged_in'):
    abort(401)
    new_entry = models.Post(request.form['title'], request.form['text'])
    db.session.add(new_entry)
    db.session.commit()
    flash('New entry was successfully posted')
    return redirect(url_for('index'))

    @app.route('/login', methods=['GET', 'POST'])
    def login():
    """User login/authentication/session management."""
    error = None
    if request.method == 'POST':
    if request.form['username'] != app.config['USERNAME']:
    error = 'Invalid username'
    elif request.form['password'] != app.config['PASSWORD']:
    error = 'Invalid password'
    else:
    session['logged_in'] = True
    flash('You were logged in')
    return redirect(url_for('index'))
    return render_template('login.html', error=error)

    @app.route('/logout')
    def logout():
    """User logout/authentication/session management."""
    session.pop('logged_in', None)
    flash('You were logged out')
    return redirect(url_for('index'))

    @app.route('/delete/', methods=['GET'])
    def delete_entry(post_id):
    """Deletes post from database."""
    result = {'status': 0, 'message': 'Error'}
    try:
    db.session.query(models.Post).filter_by(id=post_id).delete()
    db.session.commit()
    result = {'status': 1, 'message': "Post Deleted"}
    flash('The entry was deleted.')
    except Exception as e:
    result = {'status': 0, 'message': repr(e)}
    return jsonify(result)

    if __name__ == "__main__":
    app.run()
    ```

    Notice the changes in the config at the top as well since the means in which we're now accessing and manipulating the database in each view function -- via SQLAlchemy instead of vanilla SQL.

    ### Create the DB

    Run the following command to create the initial database:

    ```sh
    (env)$ python create_db.py
    ```

    ### Tests

    Finally, update the `client` fixture in the tests:

    ```python
    @pytest.fixture
    def client():
    BASE_DIR = Path(__file__).resolve().parent.parent
    app.config["TESTING"] = True
    app.config["DATABASE"] = BASE_DIR.joinpath(TEST_DB)
    app.config["SQLALCHEMY_DATABASE_URI"] = f"sqlite:///{BASE_DIR.joinpath(TEST_DB)}"

    with app.app_context():
    db.create_all() # setup
    yield app.test_client() # tests run here
    db.drop_all() # teardown
    ```

    Update the imports as well:

    ```python
    from project.app import app, db
    ```

    Ensure the tests pass:

    ```sh
    (env)$ python -m pytest

    =============================== test session starts ===============================
    platform darwin -- Python 3.10.4, pytest-7.4.2, pluggy-1.0.0
    rootdir: /Users/michael/repos/github/flaskr-tdd
    collected 6 items

    tests/app_test.py ...... [100%]

    ================================ 6 passed in 0.34s ================================
    ```

    Manually test the app as well by running the server and logging in and out, adding new entries, and deleting old entries.

    If all is well, Update the requirements file:

    ```
    Flask==3.0.0
    Flask-SQLAlchemy==3.1.1
    gunicorn==21.2.0
    pytest==7.4.2
    ```

    Commit your code, and then push the new version to Heroku!

    ## Search Page

    Let's add a search page. It will be a nice feature that will come in handy after we have a number of posts.

    ### Update *app.py*

    ```python
    @app.route('/search/', methods=['GET'])
    def search():
    query = request.args.get("query")
    entries = db.session.query(models.Post)
    if query:
    return render_template('search.html', entries=entries, query=query)
    return render_template('search.html')
    ```

    > Be sure to write a test for this on your own!

    ### Add *search.html*

    In the "templates" folder create a new file called *search.html*:

    ```sh
    (env)$ touch search.html
    ```

    Now add the following code to *search.html*:

    ```html


    Flaskr







    Flaskr





    Home

    {% if not session.logged_in %}
    log in
    {% else %}
    log out
    {% endif %}



    {% for message in get_flashed_messages() %}


    {{ message }}

    {% endfor %}



    Search:









      {% for entry in entries %} {% if query.lower() in entry.title.lower() or
      query.lower() in entry.text.lower() %}

    • {{ entry.title }}


      {{ entry.text|safe }}

    • {% endif %} {% endfor %}



    ```

    ### Update *index.html*

    Add a search button for better navigation just below `

    Flaskr

    `:

    ```html
    Search
    ```

    Test it out locally. If all is well, commit your code and update the version on Heroku.

    ## Login Required

    Currently, posts can be deleted by anyone. Let's change that so one has to be logged in before they can delete a post.

    Add the following decorator to *app.py*:

    ```python
    def login_required(f):
    @wraps(f)
    def decorated_function(*args, **kwargs):
    if not session.get('logged_in'):
    flash('Please log in.')
    return jsonify({'status': 0, 'message': 'Please log in.'}), 401
    return f(*args, **kwargs)
    return decorated_function
    ```

    Don't forget the import:

    ```python
    from functools import wraps
    ```

    > Be sure to write tests for this on your own!

    Next, add the decorator to the `delete_entry` view:

    ```python
    @app.route('/delete/', methods=['GET'])
    @login_required
    def delete_entry(post_id):
    """Deletes post from database."""
    result = {'status': 0, 'message': 'Error'}
    try:
    new_id = post_id
    db.session.query(models.Post).filter_by(id=new_id).delete()
    db.session.commit()
    result = {'status': 1, 'message': "Post Deleted"}
    flash('The entry was deleted.')
    except Exception as e:
    result = {'status': 0, 'message': repr(e)}
    return jsonify(result)
    ```

    Update the test:

    ```python
    def test_delete_message(client):
    """Ensure the messages are being deleted"""
    rv = client.get("/delete/1")
    data = json.loads(rv.data)
    assert data["status"] == 0
    login(client, app.config["USERNAME"], app.config["PASSWORD"])
    rv = client.get("/delete/1")
    data = json.loads(rv.data)
    assert data["status"] == 1
    ```

    Test it out locally again. If all is well, commit your code and update the version on Heroku.

    ## Postgres Heroku

    SQLite is a great database to use in order to get an app up and running quickly. That said, it's not intended to be used as a production grade database. So, let's move to using Postgres on Heroku.

    Start by provisioning a new `mini` plan Postgres database:

    ```sh
    (env)$ heroku addons:create heroku-postgresql:mini
    ```

    Once created, the database URL can be access via the `DATABASE_URL` environment variable:

    ```sh
    (env)$ heroku config
    ```

    You should see something similar to:

    ```sh
    === glacial-savannah-72166 Config Vars

    DATABASE_URL: postgres://zebzwxlootewbx:da5c19a66cd4765dd39aed40abb06dff10682c3213501695c4b98612de0dfac9@ec2-54-208-11-146.compute-1.amazonaws.com:5432/d77tnmeavvasm0
    ```

    Next, update the `SQLALCHEMY_DATABASE_URI` variable in *app.py* like so:

    ```python
    url = os.getenv('DATABASE_URL', f'sqlite:///{Path(basedir).joinpath(DATABASE)}')

    if url.startswith("postgres://"):
    url = url.replace("postgres://", "postgresql://", 1)

    SQLALCHEMY_DATABASE_URI = url
    ```

    So, `SQLALCHEMY_DATABASE_URI` now uses the value of the `DATABASE_URL` environment variable if it's available. Otherwise, it will use the SQLite URL.

    Make sure to import `os`:

    ```python
    import os
    ```

    Run the tests to ensure they still pass:

    ```sh
    (env)$ python -m pytest

    =============================== test session starts ===============================
    platform darwin -- Python 3.10.4, pytest-7.4.2, pluggy-1.0.0
    rootdir: /Users/michael/repos/github/flaskr-tdd
    collected 6 items

    tests/app_test.py ...... [100%]

    ================================ 6 passed in 0.32s ================================
    ```

    Try logging in and out, adding a few new entries, and deleting old entries locally.

    Before updating Heroku, add [Psycopg2](http://initd.org/psycopg/) -- a Postgres database adapter for Python -- to the requirements file:

    ```
    Flask==3.0.0
    Flask-SQLAlchemy==3.1.1
    gunicorn==21.2.0
    psycopg2-binary==2.9.9
    pytest==7.4.2
    ```

    Commit and push your code up to Heroku.

    Snce we're using a new database on Heroku, you'll need to run the following command *once* to create the tables:

    ```sh
    (env)$ heroku run python create_db.py
    ```

    Test things out.

    ## Linting and Code Formatting

    Finally, we can lint and auto format our code with [Flake8](http://flake8.pycqa.org/) and [Black](https://black.readthedocs.io/), respectively:

    ```sh
    (env)$ pip install flake8==6.1.0
    (env)$ pip install black==23.10.0
    ```

    Run Flake8 and correct any issues:

    ```sh
    (env)$ python -m flake8 --exclude env --ignore E402,E501 .

    ./create_db.py:5:1: F401 'project.models.Post' imported but unused
    ./tests/app_test.py:2:1: F401 'os' imported but unused
    ./project/app.py:2:1: F401 'sqlite3' imported but unused
    ./project/app.py:6:1: F401 'flask.g' imported but unused
    ./project/app.py:7:19: E126 continuation line over-indented for hanging indent
    ```

    Update the code formatting per Black:

    ```sh
    $ python -m black --exclude=env .

    reformatted /Users/michael/repos/github/flaskr-tdd/project/models.py
    reformatted /Users/michael/repos/github/flaskr-tdd/project/app.py
    All done! ✨ 🍰 ✨
    2 files reformatted, 4 files left unchanged.
    ```

    Test everything out once last time!

    ## Conclusion

    1. Want my code? Grab it [here](https://github.com/mjhea0/flaskr-tdd).
    1. Want more Flask fun? Check out [TestDriven.io](https://testdriven.io/). Learn how to build, test, and deploy microservices powered by Docker, Flask, and React!
    1. Want something else added to this tutorial? Add an issue to the repo.