0
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

Django REST Frameworkを理解する

Posted at

初めに

Dango REST Framework = DRFを活用するにあたり、
メリットたくさんあるけどかなり複雑だなと感じたので、
学習したないようをアウトプットしたいと思います。

目次

  • DRFとは?
  • 全体像をつかもう
  • シリアライザーを理解する
  • ビューを理解しよう
  • まとめ

DRFとは?

DRFはREST APIを作成するにあたり、多様なニーズに応えるためのDjangoのパッケージです。
Djangoだけでは満たせないたくさんの機能を補完・提供してくれます。

REST APIについても解説します。
REST APIはスマホやSPA(シングルアプリケーション)のバックエンドとしてよく利用されています。そして、URIでリソースを特定し、HTTPメソッドでリソースに対して操作するという直感的で理解しやすい設計となっていることが特徴的です。

DRFが最も得意とするのが単一リソースのCRUDを処理するREST APIです。
GET,POST,PUT,DELETEなどのHTTPメソッドに意味を持たせて、リソースをURIと組み合わせ『リソースの一覧取得』、『リソースの詳細取得』、『リソースの更新』などのAPIを作成します。

しかも、単一リソース以外の扱いもできるます。
上記の通り、単一リソースを扱うAPIの作成以外にも、

  • リソースがネストしたAPI
  • 独自のアクションを追加したAPI
  • リソースに紐づいてないAPI
    の作成もできます。

次はDRFを使うメリットについてです。
まず私が思ったのは、非常に人気が高いことから知見や解説情報が多いので、学習コストが低いなと感じました。わからないことあればだいたい出てきます。
さらに、以下のように機能がかなり充実しています。

  • Cookie認証・トークン認証・JWT認証などの認証
  • パーミッション(アクセス権)
  • フィルタリング
  • ページネーション
  • スロットリング(利用回数制限)
    など(実際まだまだあると思います)

ただ、複雑なのがデメリットです。

全体像をつかもう

Django プロジェクトの全体像

まず比較のために、Djangoプロジェクトの全体像と登場人物について説明します。図の左側から来たリクエストを、URLディスパッチャと呼ばれる Django のコアコンポーネントがハンドリングし、URLconf で登録したビュー関数を呼び出して、ビューの中でフォームやモデル、テンプレートを使って、レスポンスを作成して返し、URLディスパッチャがレスポンスを処理するという流れになっています。ミドルウェアは、ビューが呼び出される前や後で何らかの処理をするために利用されます。

image.png

image.png

DRFプロジェクト

REST API を構築するためのDRFプロジェクトでは、(HTML のフォーム要素を扱わないし、レスポンスとして画面をレンダリングした HTML を返さないので)基本的にフォームとテンプレートは使いません(後述しますが、Django を共存させる場合にはその限りではありません)。その代わり、DRFでは「シリアライザ」というコンポーネントを使います。シリアライザは入出力の JSON(およびモデルとの連携)定義とバリデーションを担当します。シリアライザはフォームと似たように使うことができます。(感想)

起点となるのは「DRF 用ビュー」です。図では「✕」で示していますが、通常の Django のビューを URLconf に登録しておくことも可能です。

image.png

image.png

シリアライザを理解する

シリアライザはフォームぽく使える

登録API、更新API、取得API のそれぞれでシリアライザの使いどころが異なります。
例えば登録 API だと、リクエストされた JSON データを元にしてシリアライザオブジェクトを作成し、シリアライザオブジェクトのバリデーションメソッドを実行し、シリアライザの永続化メソッドを実行して、最後にレスポンスオブジェクトの引数にシリアライザのデータを渡すという流れになります。
このように、ほとんどの場合でシリアライザを使うことになるため、
DRF を使うときにはシリアライザは避けて通ることができません。

image.png

シリアライザの実装イメージは次のようになります。

from rest_framework import serializers
from.models import Tweet,Comment,Like
from django.contrib.auth.models import User
from bookmarks.models import Bookmark

class TweetSerializer(serializers.ModelSerializer):
    user = UserSerializer(read_only=True)
    comments = serializers.SerializerMethodField()
    likes_count = serializers.SerializerMethodField()
    user_has_liked = serializers.SerializerMethodField()
    is_bookmarked = serializers.SerializerMethodField()

    class Meta:
        model = Tweet
        fields = ('id', 'content', 'user', 'created_at', 'comments', 'likes_count', 'user_has_liked', 'is_bookmarked')

ビューを理解しよう

DRF には『APIVie』というクラスが用意されているのですが、DRFの起点となる基底ビュークラスで、これを継承することで、さまざまな前処理と後処理をやってくれます。

  1. リクエストオブジェクトをDRF用に変換
  2. レンダラクラスとメディアタイプを決定
  3. バージョニング
  4. 認証チェック
  5. パーミッションチェック
  6. スロットリング
  7. 例外ハンドリング

2~6には代表的な処理クラスがいくつか用意されていて、デフォルトの設定を設定ファイル/settings.py/の『REST_FRAMEWORK』で上書きして全体的な設定をしたり、クラス変数をオーバーライドしてビュークラス単位での個別設定をしたりすることができます。

DRF 用のビューは大きく分けて3種類あり、用途に応じて

  1. rest_framework.views.APIView
  2. rest_framework.generics.CreateAPIView などの 汎用 APIView
  3. rest_framework.viewsets.ModelViewSet 系ビュー

のいずれかを継承して作成します。
違いはカスタム性やコード量が異なってきますが、
この後で具体的な実装を見ながら説明していきます。

ちなみに以下のようにAPIViewもDjangoのViewクラスの子クラスです。

image.png

APIViewの記述例


class LikeAPIView(APIView):
    permission_classes = [IsAuthenticated]

    def post(self, request, tweet_id):
        try:
            tweet = Tweet.objects.get(pk=tweet_id)
            like, created = Like.objects.get_or_create(user=request.user, tweet=tweet)

            if not created:
                return Response({'message': 'You already liked this tweet.'}, status=status.HTTP_200_OK)
            
            return Response({'message': 'Tweet liked successfully.'}, status=status.HTTP_201_CREATED)
        except Tweet.DoesNotExist:
            return Response({'error': 'Tweet not found'}, status=status.HTTP_404_NOT_FOUND)
        
    def delete(self, request, tweet_id):
        try:
            tweet = Tweet.objects.get(pk=tweet_id)
            like = Like.objects.filter(user=request.user, tweet=tweet)

            if not like.exists():
                return Response({'error': 'You have not liked this tweet.'}, status=status.HTTP_200_OK)
            
            like.delete()
            return Response({'message': 'Like removed successfully.'}, status=status.HTTP_200_OK)
        except Tweet.DoesNotExist:
            return Response({'error': 'Tweet not found.'}, status=status.HTTP_404_NOT_FOUND)
            

汎用APIViewの対応表

image.png

汎用APIViewの記述例


class UserCreateView(generics.CreateAPIView):
    queryset = User.objects.all()
    serializer_class = UserSerializer
    permission_classes = [permissions.AllowAny]

    authentication_classes = []

class UserDetailView(generics.RetrieveAPIView):
    queryset = User.objects.all()
    serializer_class = UserSerializer
    permission_classes = [permissions.IsAuthenticated]

    def get_object(self):
        return self.request.user

class UserViewSet(viewsets.ReadOnlyModelViewSet):
    queryset = User.objects.all()
    serializer_class = UserSerializer

ModelViewSetの対応表

image.png

ModelViewSetの記述例


class TweetViewSet(viewsets.ModelViewSet):
    queryset = Tweet.objects.all().order_by('-created_at')
    serializer_class = TweetSerializer
    permission_classes = [IsAuthenticatedOrReadOnly, IsOwnerOrReadOnly]

まとめ

DjangoにDRFの要素、コンポーネントが加わることで複雑になります。
なのでまずは、全体像を把握しましょう。
一見複雑にみえますが、新しい要素は「シリアライザ」と「DRF用ビュー」です。

「シリアライザ」は入力データのバリデーション、
内部に保持したモデルオブジェクトの永続化、
出力データの作成を担当するクラスで、
Django のフォームと同じような使い方ができます。

「DRF用ビュー」は大きく3種類あり、まるっと API 全部を実装できてしまうものもあれば、複雑なカスタム API にも対応できるものもあるので、用途に合わせて親クラスと書き方を使い分けてください。
個人的に、APIViewだけでいいのでは?とも思っています。

ありがとうございました。

0
2
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?