Flask
Knowledge Required
Basic familiarity with Python, HTML/CSS and SQL will be needed to build this website and resources can be found here:
Getting Setup
To setup your python project you should follow the guide here and install these dependencies:
- flask (Web server)
Basic Flask
To start your flask journey we can create a basic program and save it to a python file (we recommend calling it main.py
). A basic Flask program is below:
What does this all mean????
There is a lot going on here already and its only a basic program but while it might look confusing it doesn’t do too much. We have a walkthrough of each line below:
This imports the Flask library and creates a new Flask application. The __name__
variable is used to determine the path of the application. The path is the location of the application on your computer and is used to determine where to look for files such as html files. In this case we are telling Flask to look in the same directory as the main.py
file.
This is a decorator that is used to tell Flask what URL should trigger the function. In this case we are telling Flask that when the user goes to the root of the website (e.g. https://example.com/
) it should run the hello()
function. The hello()
function returns the string "Hello World!"
which is then displayed to the user. The about()
function is similar but is triggered when the user visits the /about
page.
This is the last line of the program and is used to start the Flask application. The debug=True
argument is used to tell Flask to restart the application when a file is changed. This is useful for development but should be removed when the application is deployed. This is needed to make sure the application only runs when the file is run directly and not when it is imported by another file which could cause issues.
If you run your program python main.py
you should see something like this:
If you visit http://127.0.0.1:5000
you should be able to see you webpage.
Expanding on basic Flask
Multiple Routes
With Flask you can have multiple routes in your application and each route will do something different. For example, we can add a new route that will display a different message when the user visits it. We can do this by adding the following code to our main.py
file:
This can now be accessed at http://127.0.0.1:5000/fact
. You can add as many routes as you want to your application and flask will use the string in the decorator to determine which function to run.
Wildcards
As more complex webpages are added to your application you will need to be able to access different parts of the URL. For example, you might want to have a page that displays a specific user’s profile. To do this you can use wildcards in your URL. For example, if you wanted to display a user’s profile you could do something like this:
This would allow you to change the information you display based on the URL which is especially important once you add a database to your application.
Jinja Templates
This section assumes you have basic knowledge of HTML and CSS. If you don’t know what these are you can learn more about them here.
Jinja is a templating language that is used to create HTML pages. It can be combined with Flask to create dynamic webpages. To use Jinja you need to create a folder called templates
and put all your HTML files in there. You can then use the render_template()
function to render your HTML files. For example, if you had a file called index.html
you could render it like this and have it automatically fill in some details:
Your folder structure should look like this:
- main.py
Directorytemplates/
- index.html
main.py
index.html
With Jinja it has a special syntax that uses {{ }}
to insert variables into your HTML. In this case we are inserting the name
variable into the HTML. This is useful for displaying information from a database or other sources. This can be seen from the HTML of the page which will look like this:
Splitting files and using structure
Jinja also allows you to split your HTML files into multiple files. This is useful for when you have a lot of HTML code and want to keep it organised. To do this you can use the include
function. This allows you to generate a base template that can be used for all your pages. For example, you could have a base template that contains the header and footer of your website and then include it in all your other pages. This would allow you to change the header and footer of your website in one place and have it update everywhere.
To do this you can create a file called base.html
and put it in your templates
folder. This will be the base template that all your other pages will use. You can then use the include
function to include it in your other pages. For example, if you had a page called index.html
you could do something like this:
base.html
index.html
This will render the base.html
file and then insert the index.html
file into the content
block. This allows you to have a base template that can be used for all your pages and then add extra content to each page. The final HTML will be:
You can go even further and split your HTML into multiple files. For example, you could have a file called header.html
and footer.html
and then include them in your base.html
file. This allows you to keep everything organised and make it easier to change things in the future.
base.html
header.html
footer.html
You can then use this in your page index.html
like this:
index.html
This will render the base.html
file and then insert the index.html
file into the content
block. This allows you to have a base template that can be used for all your pages and then add extra content to each page. The final HTML will be:
Fancy Jinja loops
Jinja has support for more expressive features such as loops. This allows you to loop through a list of items and display them in your HTML. For example, you could have a list of names and then loop through them and display them in your HTML. This is done using the for
loop. For example, if you had a list of names called names
you could do something like this:
This will loop through the list of names and display them in your HTML. This is useful for displaying information from a database or other sources. This can be seen from the HTML of the page which will look like this:
Other Jinja features
As a template engine, Jinja has many other features that can be used to make your life easier. These can be found in the Jinja documentation.
Databases
Databases are a very important part of any web application. They allow you to store information that can be accessed by your application. This is useful for storing information such as user accounts, blog posts, and other information. There are many different types of databases but we will be using SQLite. SQLite is a lightweight database that is easy to use and doesn’t require a server to run. This makes it a good choice for small applications. An understanding of SQL is expected for this section and you can find a tutorial here.
sqlite3
To use SQLite in Python you can use the builtin sqlite3
module. This module allows you to create a database and then run SQL queries on it. An example of how to use this module is shown below:
To create a database you can use the connect
function. This will create a new database file if it doesn’t exist or open an existing one. For example, if you wanted to create a database called users.db
you could do something like this:
This will create a new database file called users.db
in the current directory. You can then use the cursor
function to create a cursor object. This object allows you to run SQL queries on the database. For example, you could do something like this:
To run a SQL query you can use the execute
function. This function takes a string containing the SQL query and then executes it. For example, you could create a table called users
like this:
This will create a table called users
with three columns: id
, name
, and email
. The id
column is the primary key and will be automatically incremented for each new row. The name
and email
columns are both text columns. You can then use the commit
function to save the changes to the database. For example, you could do something like this:
To insert a new row into the database you can use the execute
function again. This time you will need to pass in the values that you want to insert. For example, you could insert a new user like this:
This will insert a new row into the users
table with the name John Doe
and email john.doe@example.com
and then commit the changes to the database. You can then use the fetchall
function to get all the rows from the database. For example, you could do something like this:
This will get all the rows from the users
table and then store them in the users
variable. You can then loop through the rows and display them in your HTML. For example, you could do something like this:
This will print out the rows in the users
table. This will look something like this:
To close the connection to the database you can use the close
function. For example, you could do something like this:
Combining with Flask
With our database setup we will need to combine it with our Flask application. This will allow us to access the database from our routes. To do this we will need to create a new file called database.py
. This file will contain all the code for interacting with the database. This will allow us to keep our code clean and easy to read. An example of how to do this is shown below (taken from the flask documentation here):
There is a lot of things going on here to get this database connection so let break them down below.
This function first checks if the database already exists on the requests using getattr(g, "_database", None)
. If the database doesn’t exist it adds it to the g
object using g._database = sqlite3.connect("users.db")
. The g
object is a special object in Flask that is unique for each request. This means that we can store data on the g
object and then access it later in our application.
The next thing we need to do is close the connection to the database when the request is finished. This is done using the teardown_appcontext
decorator. This decorator allows us to specify a function that is called after the request is finished. This function will then close the connection to the database. If we didn’t do this we would have a lot of open connections to the database which would eventually cause the database to crash.
Finally, we need to use the get_db
function to get the database connection in our route. This will allow us to access the database from our route. An example of this is shown above. We can then use our database connection to get the users from the database and return them in our route.
Creating a full app
With all the resources above there should be enough information to create a simple application.
Structure
With this application all the code for flask will exist in a single main.py
and there will be several webpages. The structure is as follows.
- main.py
Directorytemplates/
- base.html
- footer.html
- header.html
- index.html
- members.html
- products.html
Our main file
With flask currently all our code must exist in the same file and splitting into multiple files is outside the scope of what we are doing but useful resources on this exist in the flask documentation. Our flask app will need to contain the base code for our app to function.
We will need to setup our database which we can do when we create our db in get_db()
.
Further Resources
A copy of all the code used as a working Flask application can be found here. Below is a set of useful resources and links to help you learn more about Flask and how to use it.