文章

Flask - API 文檔化

目標

  • 安裝並配置 Flask-RESTX
  • 為現有端點添加文檔
  • 生成交互式 API 文檔頁面

步驟

  1. 準備環境

    • 繼續使用 flask_api/ 項目結構,激活虛擬環境:
      1
      2
      
      # Windows: flask_api_env\Scripts\activate
      # macOS/Linux: source flask_api_env/bin/activate
      
    • 安裝 Flask-RESTX:
      1
      
      pip install flask-restx
      
  2. 配置 Flask-RESTX

    • 修改 app/init.py,整合 Flask-RESTX:

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
      31
      32
      33
      34
      35
      36
      37
      38
      39
      40
      41
      42
      43
      44
      45
      46
      47
      48
      49
      50
      51
      52
      53
      54
      55
      56
      57
      58
      59
      60
      61
      62
      63
      64
      65
      66
      67
      68
      69
      70
      71
      72
      73
      74
      75
      76
      77
      78
      79
      80
      81
      82
      83
      84
      85
      86
      87
      88
      89
      90
      91
      92
      93
      94
      95
      96
      97
      98
      99
      100
      101
      102
      103
      104
      105
      106
      107
      108
      109
      110
      111
      112
      113
      114
      115
      
      from flask import Flask, jsonify, g
      from flask_sqlalchemy import SQLAlchemy
      from flask_marshmallow import Marshmallow
      from flask_bcrypt import Bcrypt
      from flask_restx import Api  # 新增
      import jwt
      from functools import wraps
      from .routes.v1.todos import todos_bp as todos_v1_bp
      from .routes.v1.users import users_bp as users_v1_bp
      from .routes.v1.posts import posts_bp as posts_v1_bp
      from .routes.v2.todos import todos_bp as todos_v2_bp
      from .routes.v2.posts import posts_bp as posts_v2_bp
      from .config import config_map
      from .celery_config import make_celery
      import os
      import logging
      from logging.handlers import RotatingFileHandler
      
      db = SQLAlchemy()
      ma = Marshmallow()
      bcrypt = Bcrypt()
      
      def setup_logging(app):
          if not app.debug:
              handler = RotatingFileHandler('app.log', maxBytes=10000, backupCount=3)
              handler.setLevel(logging.INFO)
              formatter = logging.Formatter(
                  '%(asctime)s %(levelname)s: %(message)s [in %(pathname)s:%(lineno)d]'
              )
              handler.setFormatter(formatter)
              app.logger.addHandler(handler)
          console_handler = logging.StreamHandler()
          console_handler.setLevel(logging.DEBUG)
          console_handler.setFormatter(logging.Formatter('%(asctime)s %(levelname)s: %(message)s'))
          app.logger.addHandler(console_handler)
          app.logger.setLevel(logging.DEBUG)
      
      def create_app():
          app = Flask(__name__)
          env = os.getenv('FLASK_ENV', 'development')
          app.config.from_object(config_map[env])
          db.init_app(app)
          ma.init_app(app)
          bcrypt.init_app(app)
          setup_logging(app)
      
          # 初始化 Flask-RESTX
          api = Api(app,
                    title='Blog API',
                    version='1.0',
                    description='A simple blog API with user and post management',
                    doc='/api/docs/')  # 文檔路徑
      
          global celery
          celery = make_celery(app)
      
          # 註冊藍圖
          app.register_blueprint(todos_v1_bp, url_prefix='/api/v1')
          app.register_blueprint(users_v1_bp, url_prefix='/api/v1')
          app.register_blueprint(posts_v1_bp, url_prefix='/api/v1')
          app.register_blueprint(todos_v2_bp, url_prefix='/api/v2')
          app.register_blueprint(posts_v2_bp, url_prefix='/api/v2')
      
          @app.errorhandler(404)
          def not_found(error):
              app.logger.error(f'404 error: {str(error)}')
              return jsonify({'error': 'Not Found', 'message': str(error)}), 404
      
          @app.errorhandler(400)
          def bad_request(error):
              app.logger.warning(f'400 error: {str(error)}')
              return jsonify({'error': 'Bad Request', 'message': str(error)}), 400
      
          @app.errorhandler(500)
          def internal_error(error):
              app.logger.critical(f'500 error: {str(error)}')
              return jsonify({'error': 'Internal Server Error', 'message': 'Something went wrong on our end'}), 500
      
          with app.app_context():
              db.create_all()
      
          return app
      
      def login_required(f):
          @wraps(f)
          def decorated_function(*args, **kwargs):
              from .models import User
              token = request.headers.get('Authorization')
              if not token:
                  abort(401, description='Missing token')
              try:
                  if token.startswith('Bearer '):
                      token = token[7:]
                  data = jwt.decode(token, app.config['SECRET_KEY'], algorithms=['HS256'])
                  user = User.query.get(data['user_id'])
                  if not user:
                      abort(401, description='Invalid token')
                  g.current_user = user
              except jwt.ExpiredSignatureError:
                  abort(401, description='Token has expired')
              except jwt.InvalidTokenError:
                  abort(401, description='Invalid token')
              return f(*args, **kwargs)
          return decorated_function
      
      def admin_required(f):
          @wraps(f)
          @login_required
          def decorated_function(*args, **kwargs):
              if not g.current_user.is_admin:
                  abort(403, description='Admin access required')
              return f(*args, **kwargs)
          return decorated_function
      
      celery = None
      
  3. 改進路由以支持文檔

    • 修改 app/routes/v1/users.py,使用 Flask-RESTX:

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
      31
      32
      33
      34
      35
      36
      37
      38
      39
      40
      41
      42
      43
      44
      45
      46
      47
      48
      49
      50
      51
      52
      53
      54
      55
      56
      57
      58
      59
      60
      61
      62
      63
      64
      65
      66
      67
      68
      69
      70
      71
      72
      73
      74
      75
      76
      77
      78
      79
      80
      81
      82
      83
      84
      85
      86
      87
      88
      89
      90
      91
      
      from flask import request, g, abort
      from flask_restx import Namespace, Resource, fields
      from ...models import User
      from ... import db, admin_required
      from ...schemas import user_schema, users_schema
      import jwt
      import datetime
      
      api = Namespace('users', description='User management operations')
      
      # 定義模型
      user_model = api.model('User', {
          'id': fields.Integer(readonly=True),
          'username': fields.String(required=True, description='The user username'),
          'password': fields.String(required=True, description='The user password'),
          'role': fields.String(description='User role (user or admin)', default='user')
      })
      
      login_model = api.model('Login', {
          'username': fields.String(required=True),
          'password': fields.String(required=True)
      })
      
      @api.route('')
      class UserList(Resource):
          @api.doc('list_users')
          @api.marshal_list_with(user_model)
          def get(self):
              """List all users"""
              users = User.query.all()
              api.app.logger.info('Fetched all users')
              return users_schema.dump(users)
      
          @api.doc('create_user')
          @api.expect(user_model)
          @api.marshal_with(user_model, code=201)
          def post(self):
              """Create a new user"""
              if not request.is_json:
                  api.abort(400, 'Request must be JSON')
              data = request.get_json()
              if 'username' not in data or 'password' not in data:
                  api.abort(400, 'Missing username or password')
              if User.query.filter_by(username=data['username']).first():
                  api.abort(400, 'Username already exists')
              user = User(username=data['username'])
              user.set_password(data['password'])
              if data.get('role') in ['user', 'admin']:
                  user.role = data['role']
              db.session.add(user)
              db.session.commit()
              api.app.logger.info(f'User created: {user.username}')
              return user_schema.dump(user), 201
      
      @api.route('/login')
      class Login(Resource):
          @api.doc('login_user')
          @api.expect(login_model)
          def post(self):
              """Login and get a JWT token"""
              if not request.is_json:
                  api.abort(400, 'Request must be JSON')
              data = request.get_json()
              if 'username' not in data or 'password' not in data:
                  api.abort(400, 'Missing username or password')
              user = User.query.filter_by(username=data['username']).first()
              if not user or not user.check_password(data['password']):
                  api.app.logger.warning(f'Failed login attempt for {data["username"]}')
                  api.abort(401, 'Invalid credentials')
              token = jwt.encode({
                  'user_id': user.id,
                  'exp': datetime.datetime.utcnow() + datetime.timedelta(hours=24)
              }, api.app.config['SECRET_KEY'], algorithm='HS256')
              api.app.logger.info(f'User logged in: {user.username}')
              return {'token': token}
      
      @api.route('/<int:user_id>')
      class UserResource(Resource):
          @api.doc('delete_user', security='jwt')
          @api.marshal_with(user_model)
          @admin_required
          def delete(self, user_id):
              """Delete a user (admin only)"""
              user = User.query.get_or_404(user_id, description='User not found')
              db.session.delete(user)
              db.session.commit()
              api.app.logger.info(f'User deleted: {user.username} by admin {g.current_user.username}')
              return {'message': 'User deleted'}, 200
      
      # 添加藍圖兼容性
      users_bp = api  # 用 Namespace 替代 Blueprint
      
    • 修改 app/routes/v1/posts.py

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
      31
      32
      33
      34
      35
      36
      37
      38
      39
      40
      41
      42
      43
      44
      45
      46
      47
      48
      49
      50
      51
      52
      53
      54
      55
      56
      57
      58
      59
      60
      61
      62
      63
      64
      65
      66
      67
      68
      69
      70
      71
      72
      73
      74
      75
      76
      77
      78
      79
      80
      81
      82
      83
      84
      85
      86
      87
      88
      89
      90
      91
      92
      93
      94
      95
      96
      97
      98
      99
      100
      101
      102
      103
      104
      105
      106
      107
      108
      
      from flask import request, g, abort
      from flask_restx import Namespace, Resource, fields
      from ...models import Post, User
      from ... import db, login_required, admin_required
      from ...schemas import post_schema, posts_schema
      from ...tasks import send_email_notification
      
      api = Namespace('posts', description='Post management operations')
      
      post_model = api.model('Post', {
          'id': fields.Integer(readonly=True),
          'title': fields.String(required=True, description='The post title'),
          'content': fields.String(required=True, description='The post content'),
          'created_at': fields.DateTime(readonly=True),
          'user_id': fields.Integer(readonly=True),
          'category': fields.String(default='general')
      })
      
      @api.route('')
      class PostList(Resource):
          @api.doc('list_posts')
          @api.marshal_list_with(post_model)
          def get(self):
              """List all posts"""
              user_id = request.args.get('user_id', type=int)
              query = Post.query
              if user_id:
                  query = query.filter_by(user_id=user_id)
              posts = query.all()
              api.app.logger.debug('Fetched posts')
              return posts_schema.dump(posts)
      
          @api.doc('create_post', security='jwt')
          @api.expect(post_model)
          @api.marshal_with(post_model, code=201)
          @login_required
          def post(self):
              """Create a new post"""
              if not request.is_json:
                  api.abort(400, 'Request must be JSON')
              data = request.get_json()
              if 'title' not in data or 'content' not in data:
                  api.abort(400, 'Missing title or content')
              post = Post(
                  title=data['title'],
                  content=data['content'],
                  user_id=g.current_user.id
              )
              db.session.add(post)
              db.session.commit()
              api.app.logger.info(f'Post created: {post.title} by {g.current_user.username}')
              send_email_notification.delay(g.current_user.id, post.title)
              api.app.logger.info(f'Queued email notification for {post.title}')
              return post_schema.dump(post), 201
      
      @api.route('/<int:post_id>')
      class PostResource(Resource):
          @api.doc('get_post')
          @api.marshal_with(post_model)
          def get(self, post_id):
              """Get a single post"""
              post = Post.query.get_or_404(post_id, description='Post not found')
              return post_schema.dump(post)
      
          @api.doc('update_post', security='jwt')
          @api.expect(post_model)
          @api.marshal_with(post_model)
          @login_required
          def put(self, post_id):
              """Update a post"""
              post = Post.query.get_or_404(post_id, description='Post not found')
              if post.user_id != g.current_user.id and not g.current_user.is_admin:
                  api.abort(403, 'You can only edit your own posts unless you are an admin')
              if not request.is_json:
                  api.abort(400, 'Request must be JSON')
              data = request.get_json()
              if 'title' in data:
                  post.title = data['title']
              if 'content' in data:
                  post.content = data['content']
              db.session.commit()
              api.app.logger.info(f'Post updated: {post.title} by {g.current_user.username}')
              return post_schema.dump(post)
      
          @api.doc('delete_post', security='jwt')
          @login_required
          def delete(self, post_id):
              """Delete a post"""
              post = Post.query.get_or_404(post_id, description='Post not found')
              if post.user_id != g.current_user.id and not g.current_user.is_admin:
                  api.abort(403, 'You can only delete your own posts unless you are an admin')
              db.session.delete(post)
              db.session.commit()
              api.app.logger.info(f'Post deleted: {post.title} by {g.current_user.username}')
              return {'message': 'Post deleted'}, 200
      
      @api.route('/all')
      class PostAll(Resource):
          @api.doc('delete_all_posts', security='jwt')
          @admin_required
          def delete(self):
              """Delete all posts (admin only)"""
              Post.query.delete()
              db.session.commit()
              api.app.logger.warning(f'All posts deleted by admin {g.current_user.username}')
              return {'message': 'All posts deleted'}, 200
      
      posts_bp = api  # 用 Namespace 替代 Blueprint
      
  4. 添加認證支持

    • app/init.py 中為 Flask-RESTX 添加 JWT 認證:
      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      
      # 在 create_app 函數內部,api 初始化後添加
      api = Api(app,
                title='Blog API',
                version='1.0',
                description='A simple blog API with user and post management',
                doc='/api/docs/',
                authorizations={
                    'jwt': {
                        'type': 'apiKey',
                        'in': 'header',
                        'name': 'Authorization',
                        'description': 'Enter "Bearer <token>"'
                    }
                })
      
  5. 運行應用

    • 運行:
      1
      
      python run.py
      
    • 訪問 http://localhost:5000/api/docs/,查看交互式 API 文檔。
  6. 測試文檔

    • 在瀏覽器中打開 /api/docs/
      • 點擊 usersposts,查看端點描述。
      • 使用 “Authorize” 按鈕輸入 Bearer <token>(從 /login 獲取)。
      • 測試 /api/v1/posts 的 POST 方法,直接在 UI 中提交數據。
  7. 作業

    • 為 v2 的 posts 路由添加 Flask-RESTX 文檔,包含 category 字段。
    • 添加一個自定義錯誤模型(例如 401 Unauthorized),並在文檔中顯示。

注意事項

  • Flask-RESTX 替換了 Blueprint,需調整路由結構。
  • @api.marshal_with 自動序列化響應,與 schemas.py 配合使用。
  • 文檔中顯示的模型字段需與實際數據結構一致。

本文章以 CC BY 4.0 授權