Skip to main content

Session Middleware for Starlette/FastAPI Applications based on Flask Session Decoding and Encoding.

This project has been archived.

The maintainers of this project have marked this project as archived. No new releases are expected.

Project description

starlette-flask

This project is archived. It achieved its goal and is no longer actively maintained. The existing PyPI package (v0.0.1) remains installable. See why this was archived below.

CI Coverage PyPI - Version PyPI - Python Version License Latest Commit

Downloads Downloads/Month Downloads/Week


Session Middleware for Starlette/FastAPI Applications based on Flask Session Decoding and Encoding.

Table of Contents

Installation

pip install starlette-flask

Usage

So what's the problem? Let's say you have a Flask application and it was live for a long time. You want to migrate to FastAPI but you don't want your users to lose their session data. And to be honest, migrating is not an easy process. You might want to take it slow and get the benefit of FastAPI features like mounting an application to another application.

Let's try to mount a Flask application to a FastAPI application.

from fastapi import FastAPI, Request, Response
from flask import Flask, jsonify, session, request
from starlette.middleware.sessions import SessionMiddleware
from a2wsgi import WSGIMiddleware

secret_key = "super-secret"


flask_app = Flask(__name__)
flask_app.config["SECRET_KEY"] = secret_key


@flask_app.get("/")
def flask_index():
    return jsonify({"message": "Hello World from Flask Application"})


@flask_app.get("/set-session")
def flask_set_session():
    session["application"] = "flask"
    session.modified = True
    return jsonify({"message": "Session set"})


@flask_app.get("/get-session")
def flask_get_session():
    return jsonify({"message": session.get("application", None)})


@flask_app.get("/delete-session")
def flask_delete_session():
    session.pop("application")
    session.modified = True
    return jsonify({"message": "Session deleted"})


@flask_app.before_request
def before_request():
    print(session.items())

@flask_app.after_request
def after_request(response):
    print(session.items())
    return response


fastapi_application = FastAPI()
fastapi_application.add_middleware(
    SessionMiddleware,
    secret_key=secret_key,
)


@fastapi_application.middleware("http")
async def starlette_add_process_time_header(request: Request, call_next):
    response = await call_next(request)
    response.headers["X-Process-Time"] = "100"
    print(response.headers)
    return response


@fastapi_application.get("/")
async def starlette_index(req: Request):
    return {"message": "Hello World from FastAPI Application"}


@fastapi_application.get("/set-session")
async def starlette_set_session(req: Request):
    req.session.update({"application": "fastapi"})
    return {"message": "Session set"}


@fastapi_application.get("/get-session")
async def starlette_get_session(req: Request):
    return {"message": req.session.get("application", None)}


@fastapi_application.get("/delete-session")
async def starlette_delete_session(req: Request):
    req.session.pop("application")
    return {"message": "Session deleted"}


app = FastAPI()
app.mount("/flask-application", WSGIMiddleware(flask_app))
app.mount("/fastapi-application", fastapi_application)

The problem here is this: If you set a session in Flask application, you can't get it from FastAPI application, and vice versa. At the same time, beyond accessing the session data, these two applications overwrite each other's session data. That's because they use different methods to sign the session data.

Since they use different methods to sign the session data, they can't decode each other's session data. What can we do? We can use starlette-flask to solve this problem.

All you need to do is this:

- from starlette.middleware.sessions import SessionMiddleware
+ from starlette_flask.middleware.sessions import SessionMiddleware

Motivation

I was migrating from Flask to FastAPI and I found out that I could use my existing Flask applications with FastAPI (thanks to a2wsgi) application.

I must tell you this: Many of my Flask applications depend on third-party Flask extensions like Flask Admin, Flask Login, and Flask-JWT-Extended

So I searched how to couple Flask and FastAPI and found a way. But there was a problem... I wasn't able to access the session data between Flask and FastAPI applications. I mean I was able to CRUD session data on the Flask side and FastAPI side but I couldn't CRUD the session data inside FastAPI that was CRUDed by Flask, so I started this discussion in the FastAPI repository. Back then I wasn't able to solve this, so I decided not to use Flask Login and Flask Admin anymore...

But you can see that the discussion didn't get any answers from March to September. It was bothering me, so I decided to solve it myself. I took a look at the source code of Flask and Starlette (backend core of FastAPI). I found that they used different methods to sign the session data and Starlette kept re-signing the session data even if it wasn't created, updated, deleted, or even read, that was the problem... I needed a custom SessionMiddleware that uses the same method as Flask to sign the session data and I did implement it.

Here are some related discussions/issues/pull requests:

Here is another related project: volfpeter/fastapi-flask-auth: Lightweight FastAPI dependencies and authenticator that uses Flask session cookies for access control. This project actually inspired me to build starlette-flask.

Check out Middleware - Starlette page to learn more about middlewares in Starlette.

Author

Disclaimer

I am not a security expert. I just wanted to share my solution to this problem. If you have any suggestions, please open an issue or a pull request.

There are also some issues and discussions about problems with the default SessionMiddleware that comes with Starlette. I believe my implementation is not perfect but either the default SessionMiddleware that comes with Starlette.

Why Archived

This project is archived as of January 2026. The reasons:

  1. Goal achieved. The middleware works and the session interoperability problem is solved.
  2. The knowledge matters more than the package. The core insight is that Flask signs session cookies using itsdangerous.URLSafeTimedSerializer with key_derivation="hmac", digest_method=hashlib.sha1, and salt=b"cookie-session". Starlette uses a different signing method. Additionally, Starlette's default SessionMiddleware re-signs the cookie on every request, even when session data hasn't changed, overwriting Flask's cookies.
  3. AI can generate this. The middleware is ~100 lines. Given the problem description above, any coding agent can produce an equivalent implementation tailored to your needs.

If you need this functionality, you can:

  • Install the existing package: pip install starlette-flask (v0.0.1, still on PyPI)
  • Copy the middleware from src/starlette_flask/middleware/sessions.py into your project
  • Ask an AI coding assistant to generate a Flask-compatible session middleware for Starlette using the signing details above

See ADR-001 for the full decision record.

License

starlette-flask is distributed under the terms of the MIT license.

Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

starlette_flask-0.1.0.tar.gz (17.1 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

starlette_flask-0.1.0-py3-none-any.whl (8.0 kB view details)

Uploaded Python 3

File details

Details for the file starlette_flask-0.1.0.tar.gz.

File metadata

  • Download URL: starlette_flask-0.1.0.tar.gz
  • Upload date:
  • Size: 17.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: uv/0.9.28 {"installer":{"name":"uv","version":"0.9.28","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for starlette_flask-0.1.0.tar.gz
Algorithm Hash digest
SHA256 bbbbb5dce8b0c11ad0b700bd97c48c5fef5ebc0b45777b58706d2b3fc36c229b
MD5 232fb81deddcfacb5e14484bddbf35b7
BLAKE2b-256 f41767fdea5f41926c7c2f61b48d41d3467348d1b0447b03ccb3bb27afbf2c30

See more details on using hashes here.

File details

Details for the file starlette_flask-0.1.0-py3-none-any.whl.

File metadata

  • Download URL: starlette_flask-0.1.0-py3-none-any.whl
  • Upload date:
  • Size: 8.0 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: uv/0.9.28 {"installer":{"name":"uv","version":"0.9.28","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for starlette_flask-0.1.0-py3-none-any.whl
Algorithm Hash digest
SHA256 fc4db74e3edf5246e5f6b1330912640ede4803a079074b6a3545a6951ef12ac9
MD5 3d79770df51255405c9781c300f55dba
BLAKE2b-256 c7267418035a1d9b4ecc72b54267f6175adba75270d074ca8bf1df4b573653b5

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page