やりたいこと
タイトルの通り、多対多のリレーションをもつModelのSerializerが返すJsonをわかりやすくまとめたい。
モデル
python
class AnimeAppearance(Model): anime = models.ForeignKey(to=Anime, on_delete=models.CASCADE) actor = models.ForeignKey(to=VoiceActor, on_delete=models.CASCADE) character = models.CharField(max_length=250, blank=True) is_mob = models.BooleanField(verbose_name="モブか", default=False) def __str__(self): return self.anime.name + " - " \ + self.actor.name + "(" + self.character + ")"
アニメ出演モデルといって、どのアニメにどの声優さんが何役で出演しているかを管理するモデルです。
アニメと声優モデルを作り別途管理しています。(カラムが今のところidとnameなのでコードは割愛します)
期待する結果
上のモデルに対するSerializerクラスを作りたいのですが、そのままだと
1出演キャラごとに1コードできるので、これをアニメごとでまとめたいです。
{ [ id:1,chara:"xxx", anime : {id:1, name:"AAA"}, actor: {id:1, name:"aaa"}], [ id:2,chara:"yyy", anime : {id:1, name:"AAA"}, actor: {id:2, name:"bbb"}], ...} ⇩ { "AAA":[ {chara:"xxx",actor: {id:1, name:"aaa"}}, {chara:"yyy", actor: {id:2, name:"bbb"}}], ...}
###書いたコード
Qiitaの記事を参考に書きました。
python
class AnimeAppearancesGroupingSerializer(serializers.ListSerializer, ABC): def to_representation(self, instance): pre = super().to_representation return[ { anime: pre(AnimeAppearance.objects.filter(anime_id=anm)) for anm in AnimeAppearance.objects.values_list("anime_id", flat=True) .order_by("anime_id").distinct() } ] @property def data(self): return ReturnDict(super().data[0], serializer=self) class AnimeAppearancesSerializer(serializers.ModelSerializer): class Meta: list_serializer_class = AnimeAppearancesGroupingSerializer model = AnimeAppearance fields = '__all__'
ABCクラスはPycharmが警告文を出すのでそれに従いました。
Viewは下の通りです。
class AnimeAppearanceView(ListAPIView): queryset = AnimeAppearance.objects.all() serializer_class = AnimeAppearancesSerializer model = AnimeAppearance filter_class = AnimeAppearancesFilter filter_backends = (OrderingFilter, DjangoFilterBackend) ordering_field = ("anime__name", "actor__name", "character") ordering = ("anime__name",)
urlは上のViewクラス.as_view()をマッピングしてます
### エラーメッセージ
まとめる前と同じurlにパラメタなしでアクセスしたら、下のエラーメッセージが出ました。
TypeError at /api/anime_appearances keys must be str, int, float, bool or None, not module
試したこと
Viewクラスのordering系のフィールドをコメントアウトしましたが変わりませんでした。
pre(AnimeAppearance.objects.filter(anime_id=anm).get("anime__name"))
に変更したら下のエラーになりました。
KeyError at /api/anime_appearances "Got KeyError when attempting to get a value for field `anime` on serializer `AnimeAppearancesSerializer`.\nThe serializer field might be named incorrectly and not match any attribute or key on the `dict` instance.\nOriginal exception text was: 'anime'."
values,values_listでも同様です。
解決方法をご教授願います。
まだ回答がついていません
会員登録して回答してみよう