资讯专栏INFORMATION COLUMN

使用swagger 生成 Flask RESTful API

printempw / 3241人阅读

摘要:指定筛选条件选择合适的状态码应答中,需要带一个很重要的字段。返回结果针对不同操作,服务器向用户返回的结果应该符合以下规范。如果状态码是,就应该向用户返回出错信息。

什么是 RESTful 什么是REST

REST(英文:Representational State Transfer,又称具象状态传输)是Roy Thomas Fielding博士于2000年在他的博士论文 中提出来的一种万维网软件架构风格,目的是便于不同软件/程序在网络(例如互联网)中互相传递信息。

REST 的核心是可编辑的资源及其集合,用符合 Atom 文档标准的 Feed 和 Entry 表示。每个资源或者集合有一个惟一的 URI。系统以资源为中心,构建并提供一系列的 Web 服务。

在 REST 中,开发人员显式地使用 HTTP 方法,对系统资源进行创建、读取、更新和删除的操作:

使用 POST 方法在服务器上创建资源

使用 GET 方法从服务器检索某个资源或者资源集合

使用 PUT 方法对服务器的现有资源进行更新

使用 DELETE 方法删除服务器的某个资源

如果一个架构符合REST原则,就可以称它为RESTful架构

RESTful API 设计定义

以下是几个RESTful API的几个概念。

资源(Resource):系统上的所有事物都被抽象为资源(一篇文章,一张照片,一段语音)

集合(Collection):一组资源的合辑称为集合(几篇文章,几张照片)

路径(Endpoint):路径又称”终点“,表示API的具体网址(每个网址代表一种资源)

那么一个设计良好的RESTful API应该遵循哪些原则呢?

协议

API与用户的通信协议总是使用HTTPs协议。

域名

应该尽量将API部署在专用域名,例如:

https://apis.gusibi.com
API地址和版本

在url中指定API版本。比如:

https://apis.gusibi.com/v1
以资源为中心设计URL

资源是RESTful API的核心元素,所有的操作都是针对特定资源进化的。而资源就是URL表示的,所以简洁、清晰、结构化的URL设计是至关重要的。
在RESTful 架构中,每个网址代表一种资源(resource),所以网址中不能有动词,只能有名词,而且所用的名词往往与数据库的表格名对应。我们来看一下 Github 的例子:

/users/:username/repos
/users/:org/repos
/repos/:owner/:repo
/repos/:owner/:repo/tags
/repos/:owner/:repo/branches/:branch
使用正确的Method

对于资源的具体操作类型,使用HTTP method 表示。
以下是常用的HTTP方法。

GET:从服务器取出资源

POST:在服务器新建一个资源

PUT:在服务器更新资源(客户端提供改变后的完整资源

PATCH:在服务器更新资源(客户端只提供改变了属性)

DELETE:从服务器删除资源

还是使用 github 的例子:

GET /repos/:owner/:repo/issues
GET /repos/:owner/:repo/issues/:number
POST /repos/:owner/:repo/issues
PATCH /repos/:owner/:repo/issues/:number
DELETE /repos/:owner/:repo
正确的过滤信息(filtering)

如果记录数量很多,服务器不能都将他们返回给用户。API应该提供参数,过滤返回结果。

下边是一些是、常见的参数。

?limit=10: 指定返回记录的数量

?offset=10:指定返回记录的开始位置

?page=2&per_page=100::指定第几页,以及每页的记录数。

?sortby=name&order=asc:指定返回结果按照哪个属性排序,以及排序顺序。

?animal_type_id=1:指定筛选条件

选择合适的状态码

HTTP 应答中,需要带一个很重要的字段:status code。它说明了请求的大致情况,是否正常完成、需要进一步处理、出现了什么错误,对于客户端非常重要。状态码都是三位的整数,大概分成了几个区间:

2XX:请求正常处理并返回
3XX:重定向,请求的资源位置发生变化
4XX:客户端发送的请求有错误
5XX:服务器端错误

常见的状态码有以下几种:

200 OK - [GET]:服务器成功返回用户请求的数据,该操作是幂等的(Idempotent)。

201 CREATED - [POST/PUT/PATCH]:用户新建或修改数据成功。
204 NO CONTENT - [DELETE]:用户删除数据成功。
400 INVALID REQUEST - [POST/PUT/PATCH]:用户发出的请求有错误,服务器没有进行新建或修改数据的操作,该操作是幂等的。
401 Unauthorized - [*]:表示用户没有权限(令牌、用户名、密码错误)。
403 Forbidden - [*] 表示用户得到授权(与401错误相对),但是访问是被禁止的。
404 NOT FOUND - [*]:用户发出的请求针对的是不存在的记录,服务器没有进行操作,该操作是幂等的。
406 Not Acceptable - [GET]:用户请求的格式不可得(比如用户请求JSON格式,但是只有XML格式)。
410 Gone -[GET]:用户请求的资源被永久删除,且不会再得到的。
422 Unprocesable entity - [POST/PUT/PATCH] 当创建一个对象时,发生一个验证错误。
500 INTERNAL SERVER ERROR - [*]:服务器发生错误,用户将无法判断发出的请求是否成功。

返回结果

针对不同操作,服务器向用户返回的结果应该符合以下规范。

GET /collection:返回资源对象的列表(数组)

GET /collection/resource:返回单个资源对象
POST /collection:返回新生成的资源对象
PUT /collection/resource:返回完整的资源对象
PATCH /collection/resource:返回完整的资源对象
DELETE /collection/resource:返回一个空文档

错误处理(Error handling)

如果出错的话,在response body 中通过 message 给出明确的信息。如果状态码是4xx,就应该向用户返回出错信息。

良好的文档

文档应该是规范的API的重要的组成部分,没有文档的API是难以给他人使用的,也是不利于维护的。

其它

使用 OAuth2.0 鉴权

尽量使用JSON作为返回的数据格式

限流

对应上述规则,我们并不能保证其它的API提供者也会遵守,特别是文档,有很大一部分API提供者给出的文档是pdf或者word文档,这是因为在API的迭代开发过程中,文档更新会比较麻烦。

swagger帮API使用者和开发者纠正了这个问题。

什么是swagger

Swagger是一个简单但功能强大的API表达工具。改框架为创建JSON或YAML格式的RESTful API 文档提供了OpenAPI规范。swagger文档可由各种编程语言处理,可以在软件开发周期中嵌入源代码控制系统中,以便进行版本管理。使用Swagger生成API,我们可以得到交互式文档,自动生成代码的SDK以及API的发现特性等。

如何编写API文档

我们可以选择使用JSON或者YAML来编写API文档。文档示例如下:

json 格式文档:

{
    "swagger": "2.0",
    "info": {
        "version": "1.0.0",
        "title": "Simple API",
        "description": "A simple API to learn how to write OpenAPI Specification"
    },
    "schemes": [
        "https"
    ],
    "host": "simple.api",
    "basePath": "/openapi101",
    "paths": {
        "/persons": {
            "get": {
                "summary": "Gets some persons",
                "description": "Returns a list containing all persons.",
                "responses": {
                    "200": {
                        "description": "A list of Person",
                        "schema": {
                            "type": "array",
                            "items": {
                                "properties": {
                                    "firstName": {
                                        "type": "string"
                                    },
                                    "lastName": {
                                        "type": "string"
                                    },
                                    "username": {
                                        "type": "string"
                                    }
                                }
                            }
                        }
                    }
                }
            }
        }
    }
}

yaml 格式文档:

swagger: "2.0"

info:
  version: 1.0.0
  title: Simple API
  description: A simple API to learn how to write OpenAPI Specification

schemes:
  - https
host: simple.api
basePath: /openapi101

paths:
  /persons:
    get:
      summary: Gets some persons
      description: Returns a list containing all persons.
      responses:
        200:
          description: A list of Person
          schema:
            type: array
            items:
              required:
                - username
              properties:
                firstName:
                  type: string
                lastName:
                  type: string
                username:
                  type: string

可以发现,yaml格式的文档比json格式的更清晰,可读性更高,推荐使用yaml格式书写文档。

swagger 官网提供了 swagger editor: http://editor.swagger.io/#/,你可以在这个编辑器中创建或导入文档,并在交互式环境中浏览它。

以下是您导入 leads.yaml 定义后的 Swagger Editor UI 外观:

右侧的显示窗格显示了格式化的文档,反映了在左侧窗格中的代码编辑器中执行的更改。代码编辑器会指出了所有格式错误。你可以展开和折叠每个窗格。

API文档的基本结构

我用一个例子来介绍下swagger文档的基本结构,这里我用yaml格式来编写文档:

swagger: "2.0"
info:
  title: Sample API
  description: API description in Markdown.
  version: 1.0.0

host: api.example.com
basePath: /v1
schemes:
  - https

paths:
  /users:
    get:
      summary: Returns a list of users.
      description: Optional extended description in Markdown.
      produces:
        - application/json
      responses:
        200:
          description: OK

上述文档包括元数据(Metadata)、Base URL、API路径(paths)三部分:

Metadata

这部分信息包括swagger 使用的版本:

swagger: "2.0"

API相关的描述信息(比如API介绍、版本等):

info:
  title: Sample API
  description: API description in Markdown.
  version: 1.0.0
Base URL

作为web API,一个很重要的信息就是用来给用户使用的 根URL,可用协议(http/https)、host地址:

host: api.example.com
basePath: /v1
schemes:
  - https

所有的API都是base URL 的相对路径 例如 /users 的API地址是 https://api.example.com/v1/users

路径(Paths)

paths 部分定义API的路径(endpoint)、支持的HTTP 请求方法

paths:  # 声明路径
  /users:  # 定义API路径
    get:   # 定义请求方式
      summary: Returns a list of users.  # 简介
      description: Optional extended description in Markdown.  # 描述
      produces:
        - application/json    # 定义 服务端response MIME types 
      responses:
        200:    # response 状态码
          description: OK

当然这只是个最简单的例子,swagger可定义的内容要比我提到的多的多。
具体详细信息可以看下 swagger 文档:https://swagger.io/docs/specification/what-is-swagger/。

当然,写完文档并不代表我们的代码就可以直接使用这份文档以及文档中的约束,swagger 还提供了 swagger-codegen:https://github.com/swagger-api/swagger-codegen。

swagger_codegen

swagger-codegen 是一个开源的代码生成工具,它包含一个模板驱动引擎,可以直接从我们定义的 swagger 文档中生成可视化的文档查看界面和API客户端。

这是一个开源的项目,地址是swagger-codegens: https://github.com/swagger-api/swagger-codegen。可以自己安装使用一下。

因为我最常用的语言是Python,所以给大家介绍一个第三方的 python 的代码生成器swagger-py-codegen:https://github.com/guokr/swagger-py-codegen

swagger_py_codegen

swagger-py-codegen的亮点是它是一个Python web framework 代码生成器,可以根据swagger 文档自动生成相应web framework 的代码,现在支持 Flask, Tornado,falcon,最新版将支持sanic。

安装

可以使用 pip 安装:

pip install swagger-py-codegen
使用

安装后使用命令如下:

swagger_py_codegen --swagger-doc api.yml example-app

可选参数有:

-s, --swagger, --swagger-doc    Swagger doc file.  [required]
-f, --force                     Force overwrite.
-p, --package                   Package name / application name.
-t, --template-dir              Path of your custom templates directory.
--spec, --specification         Generate online specification json response.
--ui                            Generate swagger ui.
-j, --jobs INTEGER              Parallel jobs for processing.
-tlp, --templates               gen flask/tornado/falcon templates, default flask.
--version                       Show current version.
--help                          Show this message and exit.

如果不指定 -tlp 参数,默认使用 flask 作为模板。
如果指定 --ui --spec 参数则会在 由-p 参数指定的目录下生成swagger UI 目录 static。

举个例子

我们这里使用 swagger-py-codegen 提供的测试文档 执行:

swagger_py_codegen --swagger-doc api.yml example-app --ui --spec

生成的代码目录结构如下

$tree
.
|__ api.yml

$ swagger_py_codegen -s api.yml example-app -p demo
$ tree (flask-demo)
.
|__ api.yml
|__ example-app
   |__ demo
   |  |__ __init__.py
   |  |__ v1
   |     |__ api
   |     |  |__ __init__.py
   |     |  |__ oauth_auth_approach_approach.py
   |     |  |__ oauth_auth_approach.py
   |     |  |__ users_token.py
   |     |  |__ users_current.py
   |     |  |__ users.py
   |     |__ __init__.py
   |     |__ routes.py
   |     |__ schemas.py
   |     |__ validators.py
   |__ requirements.txt

可以看到,这时一个简单的app框架已经生成了,其中 routes.py 是自动生成的路由,validators.py 是response和request的校验代码,schemas.py 是由文档生成的校验规则,api 目录下的各个文件是你定义的endpoint。

这时运行demo 目录下的 __init__.py 文件:

python __init__.py 

会发现 server 已经启动:

如果生成命令带上 --ui --spec,生成代码的同时也会生成swagger UI:

swagger_py_codegen --swagger-doc api.yml example-app --ui --spec

启动server后在浏览器输入地址 http://0.0.0.0:8000/static/swagger-ui/index.html#!/default/get_users_uid

可以看到直接使用的 swagger UI。

swagger-py-codegen 认证默认使用 OAuth2 认证方式,认证部分代码需要自己实现。

现在代码结构已经生成,可以安心的写逻辑代码了。

总结

这一篇主要介绍了RESTful API以及如何使用swagger编写规范的RESTful API。
最后介绍了如何使用 swagger-py-codegen 生成 web framework 的结构代码。
参考链接中的文章都非常值得一看,建议都看一下。

参考链接

REST

RESTful API 设计指南

Principles of good RESTful API Design

跟着 Github 学习 Restful HTTP API 设计

最佳实践:更好的设计你的 REST API

swagger

如何编写基于OpenAPI规范的API文档

使用 Swagger 文档化和定义 RESTful API

swagger 文档

swagger-py-codegen


最后,感谢女朋友支持。

欢迎关注(April_Louisa) 请我喝芬达

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/40623.html

相关文章

  • Flask Api 文档管理与 Swagger 上手

    摘要:众数周知,文档的编写和整理工作将花费巨大精力甚至不亚于代码的编写,因此在时间紧任务重的情况下,文档是首先被忽略的工作。是一款非常流行的文档管理交互工具,适用于在团队中的管理,以及服务组件对接。而我们目前需要的是获取文档或文件。 本文最先发布在博客:https://blog.ihypo.net/152551... Flask 是一个以自由度高、灵活性强著称的 Python Web 框架...

    Scholer 评论0 收藏0
  • Swagger 生成 PHP restful API 接口文档

    摘要:需求和背景需求为客户端同事写接口文档的各位后端同学已经在各种场合回忆了使用自动化文档工具前手写文档的血泪史我的故事却又不同因为首先来说我在公司是组负责人属于上述血泪史中催死人不偿命的客户端阵营但血泪史却是相通的没有自动化文档的日子对接口就是 需求和背景 需求: 为客户端同事写接口文档的各位后端同学,已经在各种场合回忆了使用自动化文档工具前手写文档的血泪史.我的故事却又不同,因为首先来说...

    xiaotianyi 评论0 收藏0
  • Spring Boot集成Swagger2

    摘要:前言目前互联网开发市场都流行前后台真正的分离,后台提供数据接口,前台负责请求数据并渲染。今天就介绍一款将接口文档编写和测试合并一起的集大成者,也是目前很多企业再用的一个管理工具。 前言:目前互联网开发市场都流行前后台真正的分离,后台提供数据API接口,前台负责请求数据并渲染。那么我们程序猿们在编写接口的时候,最不方便之处就是写完接口后需要进行文档的编写以及接口的测试。今天就介绍一款将接...

    liukai90 评论0 收藏0
  • SpringBoot 实战 (五) | 集成 Swagger2 构建强大的 RESTful API

    摘要:今天给你们带来集成的教程。接口返回结果不明确。这些痛点在前后端分离的大型项目上显得尤为烦躁。接口返回结果非常明确,包括数据类型,状态码,错误信息等。生成后的文件依赖如下这里使用的是的版本。另外,关注之后在发送可领取免费学习资料。 微信公众号:一个优秀的废人如有问题或建议,请后台留言,我会尽力解决你的问题。 前言 快过年了,不知道你们啥时候放年假,忙不忙。反正我是挺闲的,所以有时间写 b...

    Rindia 评论0 收藏0

发表评论

0条评论

printempw

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<