software
coding

Host Flask app on Vercel for Free

Vercel logo

What is Vercel?

First, a short introduction. Most people know Vercel as a front-end solution, however Vercel introduced many new features recently that make it a powerful tool for running small webapps for free. If you know what vercel is and why you want to deploy your project to vercel, skip here.

Vercel is a platform that enables developers to build, deploy, and manage modern web applications. Having tested it myself, I can vouch for the fact that once integrated, it is quite simple to use. With Vercel, developers can simply push their changes to their GitHub repository, and the platform will automatically redeploy the site and update the server with the new code (CI/CD Anyone?). The platform also provides a dashboard for each site, offering insights into status, recent deploys, and domain information.

Vercel dashboard example showing live preview, domains, settings

Contrary to many other services, custom domains on Vercel can be configured for free. Additional features like insights dashboard, traffic, acquisition etc. are paid features. However, if you're like me, meaning you like developing small web applications that are fine to be constrained by free-tier limitations (see table below), then Vercel is a great solution for you.

Whats the catch?

Vercel's business model mostly revolves around frontend applications, thus most of what they are offering on free tier is valid for serverless applications. While the frontend applications can access compute on its own, they do not have access to persistent memory.

Your python Vercel applications can only create/upload/edit files in the /tmp/ directory. This directory is cleaned up during each deployment, and additionally after a while when you app is 'dormant' that is, not serving any requests for a while. You can bypass this restriction if you subscribe to a paid plan.

This won't be a problem if (1) you don't need persistent storage, and (2) if your app doesn't write to the database and works on reads only.

So what is Vercel free-tier good for if it can not support a database write? Well, there are a lot of stateless applications that you could implement using flask and deploy to vercel. Simple games, simple services like convertors, userless GenAI applications etc. And this guide will be applicable to applications of the sort. If your application needs a database albeit volatile, you can have one in tmp, or if small enough, in memory.

If you're only ever going to read from your database, you can commit your database to your git repo and use it as usual.

β–² Hobby Pro Enterprise
Projects 200 Unlimited Unlimited
Deployments Created per Day 100 6000 Custom
Serverless Functions Created per Deployment Framework-dependent* ∞ ∞
Proxied Request TimeoutΒ (Seconds) 30 30 30
Deployments Created from CLI per Week 2000 2000 Custom
Vercel Projects Connected per Git Repository 3 60 Custom
Routes created per Deployment 1024 1024 Custom
Build Time per Deployment(Minutes) 45 45 45
Concurrent Builds 1 12 Custom
Disk Size (GB) 13 13 13
Cron Jobs 2* 40 100

For more, read on their blog # What does Vercel do?.

Okay, how do I deploy my flask app to Vercel then?

This is assuming you already have a Flask app up & running in your local. If not, go to # Vercel: Flask Hello World.

1. Directory Structure

Okay, you have your Flask app, and presumably, it looks something like this:

--$ git(mainline)--
my-flask-app/
β”œβ”€ app.py
β”œβ”€ helpers/
β”‚  β”œβ”€ some_helpers.py
β”‚  β”œβ”€ other_helpers.py
β”œβ”€ static/
β”‚  β”œβ”€ favicon.ico
β”‚  β”œβ”€ styles.css
β”œβ”€ templates/
β”‚  β”œβ”€ index.html
β”œβ”€ .gitignore
β”œβ”€ README.md

Let's create a new branch from HEAD, and form our new directory mapping for Vercel.

--$ git(vercel-mainline)--
my-flask-app/
β”œβ”€ api/
|  β”œβ”€ index.py # renamed from app.py
|  β”œβ”€ static/
|  β”‚  β”œβ”€ favicon.ico
β”‚  |  β”œβ”€ styles.css
|  β”œβ”€ templates/
|  β”‚  β”œβ”€ index.html
β”œβ”€ helpers/
β”‚  β”œβ”€ some_helpers.py
β”‚  β”œβ”€ other_helpers.py
β”œβ”€ vercel.json
β”œβ”€ package.json
β”œβ”€ .gitignore
β”œβ”€ README.md

Where vercel.json looks like the following, which dicatates your flask app's path from root. This snippet means that any request to your site, regardless of the path, will be rewritten to /api/index. This is useful when you have a single-page application or when you want to route all requests to a specific Serverless Function.

{
  "rewrites": [
    { "source": "/(.*)", "destination": "/api/index" }
  ]
}

P.S. Your index.py will be able to import packages as if it was at the root in this structure. No code changes required for this directory structure change.

And package.json looks like the following, and sets the node.js version to be used by Vercel. Vercel has a few supported versions, go to their documentation to see which versions are supported.

{
  "engines": {
    "node": "18.x"
  }
}

2. Create a Vercel account and connect your repositoy

Create a Vercel account, go to Projects > Import and connect your Vercel account to your github account to get access to your public or private repositories. Choose the repository of your flask app, choose vercel-mainline as the target branch and import your project.

Vercel view priompting user to import a project via github

After importing your project, vercel will attempt your initial deployment. At this point, the deployment should succeed given your repository is in the correct organizational structure introduced in (1). Congratulations, your app is up & running at <app-name>.vercel.app.

3. Environment variables

Vercel lets you define an arbitrary number of environment variables. What I like to do is running my apps on my local in debug mode. So I add the following if-else block into my main definition, and define a VERCEL environment variable.

if __name__ == '__main__':
    if os.environ.get('VERCEL', None):
        app.run(debug=False)
    else:
        app.run(debug=True)

Environment variables view on Vercel

4. Setting up a volatile database

If your app needs a database to run. Let's reiterate on the fact that Vercel does not provide any persistent data storage for your app on its free tier. But if you need a volatile database to execute tasks, you can set up your database to live on the /tmp/ directory, or completely in memory;

conn = sqlite3.connect('/tmp/test.db')
# or
conn = sqlite3.connect(':memory:')

Conclusion

This is how to get your already existing flask app running on Vercel in no time. If you have any comments, recommendations or trouble following the steps here. Please feel free to reach out πŸ‘‡

Reply on - email - mastodon

Share to Twitter