How To Get A Flask App On Heroku
by Kenneth Myers
I spent this entire past weekend trying to get my latest project onto Heroku. I found the tutorial Heroku provides wasn’t clear or, perhaps, not concise enough for me to get my app onto their site. I encountered one issue after another, most of which involved a shortage of file space or packages not installing. I hope that this guide will help people to save some time and avoid the issues that I had.
-
First, if you haven’t done so already, create an account at Heroku.
-
Next download and install the Heroku Toolbelt. This will add the Heroku commands for your terminal.
-
Type
$ heroku login
into the terminal and enter your information you used to sign up. If you are having firewall issues and/or can’t connect see this. -
Now go to your repo where the Flask app is contained. If your app uses NLTK you may want to make a copy of the repo and work on that (you may need to make a bunch of changes to your app if you’re using NLTK, or use another hosting service).
-
If you do not already have git set up in the repository, initialize git with
$ git init
. Make sure the ‘.git’ folder is located in the same directory as your Flask app. -
Original
Type
$ heroku create
. This will create a repo for your app on your Heroku account and add a remote connection from your local repo to Heroku. To see this type$ git remote -v
.Edit 3/30/16
After using this walkthrough on my second app, I came to find that Step 9 may have been wrong and if you need the conda-buildpack (numpy, scipy, etc.) then you may need to do the following in place of
heroku create
:The Conda Buildpack 2.7.X | 3.5.X
3.5.X
$ heroku create --buildpack https://github.com/buildingspeak/conda-buildpack.git
2.7.X
$ heroku create --buildpack https://github.com/kennethreitz/conda-buildpack.git
-
Make sure that in your
flask_app.py
file (the one that executes the app) this is at the bottom instead of whatever you might have used earlier:if __name__ == '__main__': app.debug = True port = int(os.environ.get("PORT", 5000)) app.run(host='0.0.0.0', port=port)
-
Next you will add some crucial files to your repo:
-
Procfile : name this file exactly like this, WITHOUT any ‘.txt’ extension. Inside you should have:
web: python flask_app.py
You should replace flask_app.py with the name of your flask app. Note that I dont have gunicorn because I don’t use that as the webserver and am not familiar with it.
-
requirements.txt : This file and the following two HAVE the ‘.txt’ extension. In this file you will list the packages that pip should install for you. Most tutorials recommend that you should do
$ pip freeze > requirements.txt
but I think you should only do this if you are working on a virtual environment that only has the necessary packages installed. Otherwise that command will add every package you have installed which will rapidly take up your apps size. For mine I only added the packages that I knew were necessary and pip took care of their dependencies. Here’s an example:Flask==0.10.1 Jinja2==2.8 matplotlib==1.5.1 mpld3==0.2
-
runtime.txt : This file is necessary if you use a python installation that is NOT 2.7.X (3.5.1 for example). Inside you would write:
python-3.5.1
-
conda-requirements.txt : In this file you list packages that require non-python dependencies (for excample C/C++). If later you find that you are going over your build size when you try pushing, look to see if Heroku is trying to build
MKL
. MKL is an Intel package that speeds up numpy/scipy computations. However, the file size is relatively huge and it is not completely necessary to run an app. If you need, addnomkl
to this file. Here’s an example:nomkl numpy pandas scipy scikit-learn
-
-
Edit 3/30/16
After using this walkthrough on my second app, I am no longer sure if this step is necessary if you already created your app with the conda-buildpack in step 6. Doing this step won’t hurt (I still did it) but it may not be needed.
Original
Now you’ve got all the necessary files. However, there are still a few things you have to do before you’re app will build. You have to add the build packs:
-
The Python Buildpack : This will set the current buildpack to python and ensure that the it builds properly.
$ heroku buildpacks:set heroku/python
-
The Conda Buildpack 2.7.X | 3.5.X : I use python 3.5 and I used the 3.5.X buildpack linked above and it worked for me. A friend who uses 2.7.X used its respective pack and it worked for him.
3.5.X
$ heroku config:add BUILDPACK_URL=https://github.com/buildingspeak/conda-buildpack.git
2.7.X
$ heroku config:add BUILDPACK_URL=https://github.com/kennethreitz/conda-buildpack.git
-
Also, I do not recall if this is necessary but both and me and my friend added this to the apps config variables and it didn’t break anything (as far as we can tell) :
$ heroku config:add LD_LIBRARY_PATH=/app/.heroku/vendor/lib/atlas-base/atlas:/app/.heroku/vendor/lib/atlas-base:/app/.heroku/vendor/lib/
If anything it seems to be adding the path to Atlas, the alternative to MKL that I removed earlier in the ‘conda-requirements.txt’ step.
-
-
You’re almost there! Do the following:
$ git add . $ git commit -m 'some message' $ git push heroku master
Watch it build the ‘slug’ and make sure there aren’t any errors. As I said before, I had many problems at this step. You may find that you need to clear your Heroku cache which can be done if you add the Heroku repo plugin. I also said before that I could not get NLTK to work. This is because NLTK’s corpora is enormous, over 2GB. I had read of a solution to add the TextBlob corpora using the
post-compile
method but when I tried to apply this to NLTK, it still didn’t work (it still tried to include it in the slug). The only solution for me was to load pickles of my preprocessing into my app but another alternative would be to put everything on AWS (a project for another day). -
Once it has built properly, deploy the app:
$ heroku ps:scale web=1 $ heroku open
- That’s it! If for some reason it won’t load, type
$ heroku logs
to see where the error occured. One last useful command is$ heroku ps
to check the status of the dyno.
If you get any other errors then either this tutorial has gotten out of date or they might be more specific to your own code. Good luck!
Subscribe via RSS