arrow left
Back to Developer Education

How to Handle File Uploads with Flask

How to Handle File Uploads with Flask

Flask is a lightweight or micro web framework built with Python that helps in creating web applications. It provides useful tools and features that make building web applications easier. Flask is extensible and doesn’t force a particular structure or require complicated boilerplate code before getting started. It gives developers flexibility. <!--more-->

Introduction

One important feature in web applications is the ability to let users upload files. These files could be pictures, PDF, audio CSV, etc. In this article, we will look at how to set up a basic flask app that will allow users to upload files.

Prerequisites

Going through this guide, it is assumed that the reader has a basic knowledge of Python programming language, HTML, and they must have a fundamental knowledge of flask; even though this guide will be beginner-friendly.

In this guide, we will be using Python 3, and VS Code text editor you can download vscode and Python

Goal

We will be building a flask app that will enable users to upload files to a server. At the end of this guide, the reader will be familiar with:

  • Creating a virtual environment
  • Activating a virtual environment
  • Setting up a flask app
  • Enabling file uploads

Python virtual environment

A virtual environment is an isolated environment for Python projects. There is a module created by Python called venv which gives a developer a unique environment that enables the installation of all packages that are unique to a particular project.

The virtual environment doesn’t change the default Python version or default packages installed in a system, instead, it gives you freedom from the interference of other packages installed in the system. This makes it easy to run any Python project on any computer irrespective of the Python version or packages installed in the system.

How to create a virtual environment

The process of creating a virtual environment differs based on the operating system. In this guide, we will look at the process in the context of a windows operating system.

Follow the link to see how it's done on a Mac and on a Ubuntu.

To start, on a Windows device open PowerShell and make a directory using the command below:

mkdir

Get into the new directory using the cd directory-name then install the virtual environment using the command:

pip install virtualenv

Then create the virtual environment using the command:

virtualenv myenv

Note that myenv is the name of my virtual environment it can be any name you wish. Next, activate the virtual environment using the command:

myenv/Scripts/activate

If you are using the command-line interface (CMD) your command will be as below:

myenv\Scripts\activate.bat

Creating our project

After activating our virtual environment, we can now create our project. To do that, we will make a new directory for the project.

Use the command below:

mkdir tutorial

NOTE: tutorial is my project's name. You can give yours any name you like. To build a flask application, we must first install flask.

To do that, we will use the command below:

pip install flask 

After the installation, we will create a new file with the name app.py, update the file with the code below:

from flask import Flask

app = Flask(__name__)


@app.route('/')
def index():
    return"hello world"

if __name__==('__main__'):
    app.run(debug=True)

From the code above we are importing flask from the flask library we installed.

The @app.route is doing the routing for us.

The index() is our view function which will return our page content to the browser.

The if statement returns the app.run, which will enable us to run our app then refresh our page whenever we save changes. To run our app we run the command below on our terminal.

python app.py

Note that app.py is the name of my app yours can be different. If everything goes well you will have a result like the one shown below.

App running

To upload files, we will use the WTforms and the flask-uploads libraries. To work with these libraries we need to install them.

Do that with the command below:

pip install flask_wtf, WTForms
pip install flask-uploads

After the installation, we will create a file field, by updating the code to the one below:

from flask import Flask, render_template
from flask_wtf import FlaskForm
from wtforms import FileField

app = Flask(__name__)

class MyForm(FlaskForm):
    image = FileField('image')

@app.route('/')
def index():
    form = MyForm()
    return render_template('index.html')

if __name__==('__main__'):
    app.run(debug=True)

From the code above, we start by importing FlaskForm from flask_wtf and FileField from wtforms. Next, we created a class for our form as Myform image is the file field our image files will be saved to. We call our Form class in our index function. We changed our render to render template.

This is also a flask library used for rendering HTML templates. From the code we rendered index.html. When we use render_template in Flask we create a folder called templates where we store the HTML files. Now let us create the HTML template we are rendering, inside our templates folder.

Update the HTML file with the code below:

!doctype html>
<html>
  <head>
    <title>File Upload</title>
  </head>
  <body>

    <form action="/" method="POST" enctype="multipart/form-data">
        {{ form.csrf_token }}
        {{ form.image }}
        <button type="submit">upload</button>
    </form>
  </body>
</html>

From the code above, our form takes a method POST because we will be posting a file. The csrf_token is a built-in function that handles security for us, then we call our form field we created in our Form Class using form.image. Now we can run our app using python app.py. If everything is correct you will get a runtime error like in the image below.

RuntimeError

This error occurs whenever you try to use a csrf_token without adding a secret_key to your project file. Let's add a secret key to our code.

Update your code to the one below:

from flask import Flask, render_template
from flask_wtf import FlaskForm
from wtforms import FileField

app = Flask(__name__)

app.config['SECRET_KEY'] = 'mysecretkey'

class MyForm(FlaskForm):
    image = FileField('image')

@app.route('/')
def index():
    form = MyForm()
    return render_template('index.html')

if __name__==('__main__'):
    app.run(debug=True)

The secret_key can be anything you want.

Let's update our code to the one below:

from flask import Flask, render_template
from flask_wtf import FlaskForm
from wtforms import FileField

app = Flask(__name__)

app.config['SECRET_KEY'] = 'mysecretkey'

class MyForm(FlaskForm):
    image = FileField('image')

@app.route('/')
def index():
    form = MyForm()
    return render_template('index.html', form = form)

if __name__==('__main__'):
    app.run(debug=True)

Our page should now look like the picture below:

App running

From the code above, form=form is parsed so that our form can be displayed on our HTML page. If we try to upload an image, we will encounter another error as shown below:

Method Not Allowed

This error is often thrown when we don't specify a method to our route. To solve this, we will add the code below to our route.

@app.route('/', methods=['GET', 'POST'])

After adding the above code, our upload will work but it won't be saved because we didn't give it a path to save to. This is where flask uploads come into play.

Let's import flask-uploads using the command:

from flask_uploads import configure_uploads, IMAGES, UploadSet

configure_uploads enables us to set the path for the image to be saved, IMAGES is the file type we are uploading.

We will update our code with: app.config['UPLOADED_IMAGES_DEST'] = 'uploads/images this will set the file path where the images will be saved, images = UploadSet('images', IMAGES) and configure_uploads(app, images) saves the file extension and configure the uploads.

if form.validate_on_submit():
        filename = images.save(form.image.data)
        return f'Filename: {filename}'
    return render_template('index.html', form = form)

The above snippet will validate and save our image file.

Our final code will look like the one below:

from flask import Flask, render_template
from flask_wtf import FlaskForm
from wtforms import FileField
from flask_uploads import configure_uploads, IMAGES, UploadSet

app = Flask(__name__)

app.config['SECRET_KEY'] = 'thisisasecret'
app.config['UPLOADED_IMAGES_DEST'] = 'uploads/images'

images = UploadSet('images', IMAGES)
configure_uploads(app, images)


class MyForm(FlaskForm):
    image = FileField('image')

@app.route('/', methods=['GET', 'POST'])
def index():
    form = MyForm()
    if form.validate_on_submit():
        filename = images.save(form.image.data)
        return f'Filename: {filename}'
    return render_template('index.html', form = form)

if __name__==('__main__'):
    app.run(debug=True)

After uploading a file, the file name will be return as seen in the image below:

File Uploaded

Conclusion

Now we can upload images. To upload other types of files all we need to do is to import them through flask upload, configure their destination path, and specify their file extension.

Learn more about flask-uploads by clicking the link in the further reading section. Link to project Github Repo.

Happy coding!

Further reading


Peer Review Contributions by: Jerim Kaura

Published on: Jul 29, 2021
Updated on: Jul 12, 2024
CTA

Cloudzilla is FREE for React and Node.js projects

Deploy GitHub projects across every major cloud in under 3 minutes. No credit card required.
Get Started for Free