Как настроить REST API во Flask за 5 шагов
Существует множество способов создания REST API, наиболее распространенным из которых является приложение Django с DRF. Другие люди пробуют FastAPI.
Но если вы используете приложение на основе Flask, я недавно попробовал библиотеку Flask-RESTX, которая включает в себя несколько замечательных функций:
- Документация Swagger (черт возьми, да!)
- Маршаллинг ответа
- Cинтаксический анализ запросов
- Обработка ошибок, ведение журнала и поддержка blueprint. Аккуратная интеграция с Flask.
В этой демонстрации я покажу вам, как настроить быстрый REST API с документацией Swagger, синтаксическим анализом запросов и простым форматированием ответа.
Начнем с инициализации blueprint и определения объекта api в новом модуле. Я назвал это как api.py
.
blueprint = Blueprint("api", __name__, url_prefix="/api/v1")
api = Api(
blueprint,
version="1.0",
title="Mini REST API",
description="A mini REST API",
)
ns = api.namespace("items", description="Item operations")
api.add_namespace(ns)
Flask-RESTX поддерживает Flask Blueprint, и их очень просто реализовать.
Мое приложение обслуживается по http://localhost:5000
, но мой базовый URL-адрес API будет http://localhost:5000/api/v1
. На этой же странице вы можете найти документацию Swagger.
Далее напишем базовые модели. Мой пример API будет управлять объектами Items и Details, поэтому мне нужно написать модели, которые будут отвечать за их представление в стандартном ответе API.
detail_model = api.model("Detail", {"id": fields.Integer, "name": fields.String})
item_model = api.model(
"Item",
{
"id": fields.Integer,
"name": fields.String,
"details": fields.List(fields.Nested(detail_model)),
},
)
Идея написания моделей заключается в использовании маршалинга ответов Flask-RESTX, поэтому независимо от того, масштабируются ли наши объекты, ответ всегда будет таким, каким мы его документируем в наших моделях. Flask-RESTX включает в себя множество инструментов для этого, таких как переименование атрибутов, сложных, настраиваемых и вложенных полей и т.д.
Последний шаг настройки - написание парсера запроса.
item_parser = api.parser()
item_parser.add_argument("id", type=int, location="form")
item_parser.add_argument("name", type=str, location="form")
detail_parser = api.parser()
detail_parser.add_argument("id", type=int, location="form")
detail_parser.add_argument("name", type=str, location="form")
Таким же образом, как и раньше, мы используем синтаксический анализатор запросов Flask-RESTX для чтения и проверки значений, которые мы ожидаем получить в наших конечных точках. В этом случае я планирую реализовать два объектных API, которые будут добавлять элементы к нашим объектам базы данных. (Наша база данных - это простой объект в памяти 😅)
memory_object = [
{
"id": 1,
"name": "Item 1",
"details": [
{"id": 1, "name": "Detail 1"},
{"id": 2, "name": "Detail 2"},
],
}
]
Пришло время реализовать наши API. Первый API, который я хочу создать, - это тот, который управляет элементами. Я назову это ItemApi
, и маршрут будет /
, что означает корень пространства имен items
.
@ns.route("/")
class ItemsApi(Resource):
"""
API for handling the Item list resource
"""
@api.response(HTTPStatus.OK.value, "Get the item list")
@api.marshal_list_with(item_model)
def get(self) -> list[Item]:
"""
Returns the memory object
"""
return memory_object
@api.response(HTTPStatus.OK.value, "Object added")
@api.expect(item_parser)
def post(self) -> None:
"""
Simple append something to the memory object
"""
args = item_parser.parse_args()
memory_object.append(args)
Это активирует две конечные точки:
- GET
/api/v1/items/
- Список item_model - POST
/api/v1/items/
- Созданиеitem_parser
Все декораторы предоставляются Flask-RESTX. Класс HTTPStatus
предоставляется модулем http
. Довольно просто, а?
Этот класс будет управлять одним ресурсом элемента. Итак, чтобы получить его данные и добавить детали, нам понадобится следующая реализация:
@ns.route("/<int:item_id>")
class ItemApi(Resource):
"""
API for handling the single Item resource
"""
@api.response(HTTPStatus.OK.value, "Get the item list")
@api.response(HTTPStatus.BAD_REQUEST.value, "Item not found")
@api.marshal_with(item_model)
def get(self, item_id: int) -> Item:
"""
Returns the memory object
"""
try:
return self._lookup(item_id)
except StopIteration:
return api.abort(HTTPStatus.BAD_REQUEST.value, "Item not found")
def _lookup(self, item_id):
return next(
(item for item in memory_object if item["id"] == item_id),
)
@api.response(HTTPStatus.NO_CONTENT.value, "Object added")
@api.response(HTTPStatus.BAD_REQUEST.value, "Item not found")
@api.expect(detail_parser)
def post(self, item_id: int) -> None:
"""
Simple append details to the memory object
"""
args = item_parser.parse_args()
try:
if item := self._lookup(item_id):
item["details"].append(args)
return None
except StopIteration:
return api.abort(HTTPStatus.BAD_REQUEST.value, "Item not found")
Это позволит включить еще две конечные точки:
- GET
/api/v1/items/<item_id>
- Eдиный ресурсitem_model
- POST
/api/v1/items/<item_id>
- Созданиеdetail_parser
Чтобы завершить наше приложение, вам нужно только импортировать модуль app.py
и зарегистрировать Blueprint.
from api import blueprint
app = Flask(__name__) # This line already exists
app.register_blueprint(blueprint)