Skip to main content

Flask extension designed to effortlessly validate requests with Pydantic based on standard Python type hints.

Project description

CI pypi codecov license Downloads

About

flask_typed_routes is a Flask extension designed to effortlessly validate requests with Pydantic based on standard Python type hints.

Documentation: https://rmoralespp.github.io/flask_typed_routes/

Features

  • 🎯 Type Safety: Automatically validates requests using Python type hints.
  • 🔌 Easy Integration: Simple extension for validating Flask routes.
  • ⚠️ Error Handling: Clear and automatic responses for validation failures.
  • Autocomplete: Editor integration with comprehensive suggestions.
  • ⚙️ Validation Modes: Supports automatic validation for all routes and manual validation for specific routes using decorators.
  • 📖 OpenAPI Support: Automatically generates an OpenAPI schema, ensuring clear documentation and seamless integration with OpenAPI tools.

Requirements

  • Python 3.10+
  • Pydantic 2.0+
  • Flask

Installation

To install flask_typed_routes using pip, run the following command:

pip install flask_typed_routes

Getting Started

This tool offers comprehensive validation for various types of request parameters, including Path, Query, Body, Header, and Cookie parameters.

Example of a simple Flask application using flask_typed_routes:

Create a file items.py with:

import typing as t

import annotated_types as at
import flask
import flask_typed_routes as ftr
import pydantic

app = flask.Flask(__name__)
ftr.FlaskTypedRoutes(app)

Skip = pydantic.NonNegativeInt # custom Pydantic type
Limit = t.Annotated[int, at.Ge(1), at.Le(100)] # custom Annotated type

@app.get('/items/<user>/')
def read_items(user: str, skip: Skip = 0, limit: Limit = 10):
    # Parameters not included in the "path" are automatically treated as "query" parameters.
    data = {
        'user': user,
        'skip': skip,
        'limit': limit,
    }
    return flask.jsonify(data)

Run the server with:

flask --app items run --debug

Open your browser and go to http://127.0.0.1:5000/items/myuser/?skip=20 You will see the JSON response as:

{
  "limit": 10,
  "skip": 20,
  "user": "myuser"
}

Validation: Open your browser and go to http://127.0.0.1:5000/items/myuser/?skip=abc You will see the JSON response with the error details because the skip parameter is not an integer:

{
  "errors": [
    {
      "input": "abc",
      "loc": [
        "query",
        "skip"
      ],
      "msg": "Input should be a valid integer, unable to parse string as an integer",
      "type": "int_parsing",
      "url": "https://errors.pydantic.dev/2.9/v/int_parsing"
    }
  ]
}

Example with Pydantic Models

You can also use Pydantic models to validate request data in Flask routes. Now let's update the items.py file with:

import pydantic
import flask
import flask_typed_routes as ftr

app = flask.Flask(__name__)
ftr.FlaskTypedRoutes(app)


class Item(pydantic.BaseModel):
    name: str
    description: str = None
    price: float


@app.post('/items/')
def create_item(item: Item):
    return flask.jsonify(item.model_dump())


@app.put('/items/<item_id>/')
def update_item(item_id: int, item: Item):
    return flask.jsonify({'item_id': item_id, **item.model_dump()})

Using Flask Blueprints

You can also use flask_typed_routes with Flask Blueprints.

Now let's update the items.py file with:

import flask
import flask_typed_routes as ftr

app = flask.Flask(__name__)
ftr.FlaskTypedRoutes(app)
blp = flask.Blueprint('items', __name__, url_prefix='/v2')


@blp.get('/items/')
def read_items(skip: int = 0, limit: int = 10, country: str = 'US'):
    data = {'skip': skip, 'limit': limit, 'country': country}
    return flask.jsonify(data)


app.register_blueprint(blp)

Using Flask Class-Based Views

You can also use flask_typed_routes with Flask Class-Based Views.

Now let's update the items.py file with:

import flask
import flask.views
import flask_typed_routes as ftr

app = flask.Flask(__name__)
ftr.FlaskTypedRoutes(app)


class UserProducts(flask.views.View):

    def dispatch_request(self, user: str, skip: int = 0, limit: int = 10):
        data = {'user': user, 'skip': skip, 'limit': limit}
        return flask.jsonify(data)


class UserOrders(flask.views.MethodView):

    def get(self, user: str, skip: int = 0, limit: int = 10):
        data = {'user': user, 'skip': skip, 'limit': limit}
        return flask.jsonify(data)


app.add_url_rule('/products/<user>/', view_func=UserProducts.as_view('user_products'))
app.add_url_rule('/orders/<user>/', view_func=UserOrders.as_view('user_orders'))

Interactive API docs

You can OpenApi schema generated by flask_typed_routes with any OpenApi tools to generate interactive API docs for your Flask application. In this example we will use the swagger-ui-py library.

pip install swagger-ui-py  # ignore if already installed
import flask
import pydantic
import swagger_ui

import flask_typed_routes as ftr

app = flask.Flask(__name__)
app_ftr = ftr.FlaskTypedRoutes(app)
swagger_ui.api_doc(app, config_rel_url=app_ftr.openapi_url_json, url_prefix=app_ftr.openapi_url_prefix)


class Item(pydantic.BaseModel):
    name: str
    description: str = None
    price: float


@app.get('/items/<user>/')
def read_items(user: str, skip: int = 0, limit: int = 10):
    data = {'user': user, 'skip': skip, 'limit': limit}
    return flask.jsonify(data)


@app.post('/items/')
def create_item(item: Item):
    return flask.jsonify(item.model_dump())


@app.put('/items/<item_id>/')
def update_item(item_id: int, item: Item):
    return flask.jsonify({'item_id': item_id, **item.model_dump()})


@app.delete('/items/<item_id>/')
def remove_item(item_id: int):
    return flask.jsonify({'item_id': item_id})

Open your browser and go to http://127.0.0.1:5000/docs/

OpenApi Example

Create item endpoint:

OpenApi Example

Read Items endpoint:

OpenApi Example

Documentation

For more detailed information and usage examples, refer to the project documentation

Development

To contribute to the project, you can run the following commands for testing and documentation:

Running Unit Tests

Install the development dependencies and run the tests:

(env)$ pip install -r requirements-dev.txt  # Skip if already installed
(env)$ python -m pytest tests/
(env)$ python -m pytest --cov # Run tests with coverage

Building the Documentation

To build the documentation locally, use the following commands:

(env)$ pip install -r requirements-doc.txt # Skip if already installed
(env)$ mkdocs serve # Start live-reloading docs server
(env)$ mkdocs build # Build the documentation site

License

This project is licensed under the MIT license.

Project details


Download files

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

Source Distribution

flask_typed_routes-0.2.2.tar.gz (18.8 kB view details)

Uploaded Source

Built Distribution

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

flask_typed_routes-0.2.2-py3-none-any.whl (18.2 kB view details)

Uploaded Python 3

File details

Details for the file flask_typed_routes-0.2.2.tar.gz.

File metadata

  • Download URL: flask_typed_routes-0.2.2.tar.gz
  • Upload date:
  • Size: 18.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.9.21

File hashes

Hashes for flask_typed_routes-0.2.2.tar.gz
Algorithm Hash digest
SHA256 66bf632c6cfc43e2394720aefe7e17a1c11f6ae16d8133548f6c4b7bed538678
MD5 88cb12f3a3faa360c9532db8a5849617
BLAKE2b-256 c44c5d83a9b96645408814d1bdfda052260d82c6cb811ed2c713357e49d7e40a

See more details on using hashes here.

File details

Details for the file flask_typed_routes-0.2.2-py3-none-any.whl.

File metadata

File hashes

Hashes for flask_typed_routes-0.2.2-py3-none-any.whl
Algorithm Hash digest
SHA256 9e1c5486748e9a6ab8ffb3bec03a6036eeeb19afbd5d91728e48047a9f1c6474
MD5 70250ebc5ce07a5f57d58373d2e5db57
BLAKE2b-256 3162a18598536d832a5b8c5f66da73b2e9f86490bd521a38ec5b7e652f30a214

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