Using the Python Runtime with Vercel Functions
Use the Python runtime to run ASGI (Asynchronous Server Gateway Interface) and WSGI (Web Server Gateway Interface) applications on Vercel. The Python Framework Presets work with FastAPI, Flask, Django, and other Python web frameworks.
Vercel detects your framework automatically when it finds a matching dependency in
requirements.txt, pyproject.toml, or Pipfile. Define a supported Python
entrypoint so Vercel can load your application.
A Python entrypoint is the file and top-level variable that Vercel loads as the Vercel Function handler. Vercel looks for a Python entrypoint in these locations:
app.py,index.py,server.py,main.py,wsgi.py, orasgi.py- the same filenames inside
src/,app/, orapi/ - the module path configured with
tool.vercel.entrypointinpyproject.toml
The entrypoint file must define one of these top-level names:
appfor most ASGI or WSGI frameworks, including FastAPI and Flaskapplicationfor Django and other WSGI applicationshandlerfor Python serverless functions that useBaseHTTPRequestHandler
To point Vercel to an app in a custom module, set tool.vercel.entrypoint to a
Python module:variable value:
[tool.vercel]
entrypoint = "my_package.api:app"The tool.vercel.entrypoint value tells Vercel to load the app variable
from my_package/api.py.
Vercel still supports [project.scripts] app = "module:variable" for existing
projects. Use tool.vercel.entrypoint for new projects.
Here's a FastAPI example:
from fastapi import FastAPI
app = FastAPI()
@app.get("/")
def home():
return {"message": "Hello from Python on Vercel"}
@app.get("/api/items/{item_id}")
def read_item(item_id: int):
return {"item_id": item_id}[project]
name = "my-python-api"
version = "0.1.0"
requires-python = ">=3.12"
dependencies = [
"fastapi>=0.117.1",
]To deploy a Python API alongside a frontend such as a Next.js app within the same project, use Services.
For framework-specific setup guides, see:
Set the Python version for your project with pyproject.toml,
.python-version, or Pipfile.lock. For step-by-step instructions, see
Set the Python version for your Vercel project.
If the required Python version is not defined or not supported, Vercel uses the default version. The available versions are:
- 3.12 (default)
- 3.13
- 3.14
Define dependencies in pyproject.toml (with or without a uv.lock),
requirements.txt, or a Pipfile with a corresponding Pipfile.lock.
fastapi==0.117.1An example requirements.txt file that defines FastAPI as a dependency.
Vercel Functions support streaming responses when using the Python runtime. This lets you send parts of a response as they become ready.
By default, Python Vercel Functions include all files from your project that are reachable at build time. There is no automatic tree-shaking for Python.
Make sure your pyproject.toml or requirements.txt only lists packages
necessary at runtime. Explicitly exclude files you don't need to keep bundles
small and avoid hitting size limits.
Python functions have a maximum uncompressed bundle size of 500 MB. See the bundle size limits.
To exclude unnecessary files (tests, static assets, test data), configure
excludeFiles in vercel.json under the functions key. The pattern is a
glob relative to your
project root.
{
"$schema": "https://openapi.vercel.sh/vercel.json",
"functions": {
"api/**/*.py": {
"excludeFiles": "{tests/**,__tests__/**,**/*.test.py,**/test_*.py,fixtures/**,__fixtures__/**,testdata/**,sample-data/**,static/**,assets/**}"
}
}
}Exclude common development and static folders from all Python functions to stay under the 500 MB bundle limit.
Python uses the current working directory when you pass a relative path to open(). The working directory is the base of your project, not the directory containing the file.
You can also place .py files inside an /api directory. Each file that
defines a handler (inheriting from BaseHTTPRequestHandler) or an ASGI/WSGI
app becomes a separate Vercel Function.
from http.server import BaseHTTPRequestHandler
class handler(BaseHTTPRequestHandler):
def do_GET(self):
self.send_response(200)
self.send_header('Content-type','text/plain')
self.end_headers()
self.wfile.write('Hello, world!'.encode('utf-8'))
returnWas this helpful?