質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

88.92%

Django REST frameworkで実装したAPIが400エラーを返す

解決済

回答 1

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 247

shun173

score 4

前提・実現したいこと

DjangoのモデルにアクセスするAPIを作成しました。このAPIにpostメソッドでテストを実行すると400エラーが返ってきてしまいます。getメソッドによるテストは上手くいきました。また、ブラウザ上でpostを実行したときは上手くいきました。

発生している問題・エラーメッセージ

FAIL: test_post (ecapp.tests.test_api.ProductApiTest)
----------------------------------------------------------------------
Traceback (most recent call last):
  File "C:\Users\lion-\Desktop\portfolio\ecapp\tests\test_api.py", line 82, in test_post
    self.assertEqual(response.status_code, status.HTTP_201_CREATED)
AssertionError: 400 != 201

該当のソースコード

test_api.py

from django.core.files import File
from django.core.files.uploadedfile import SimpleUploadedFile
from django.test import TestCase, Client
from django.test.client import encode_multipart
from django.urls import reverse
from django.utils.timezone import localtime
from faker import Faker
from rest_framework import status
from rest_framework.test import APIRequestFactory, force_authenticate, APIClient, APITestCase

from ..models import Product
from ..views import ProductViewSet
from .factories import ProductFactory, UserFactory


class ProductApiTest(APITestCase):
    def test_post(self):
        user = UserFactory()
        view = ProductViewSet.as_view({'post': 'create'})
        filename = 'test_img'
        file = File(open('static/default_icon.jpg', 'rb'))
        uploaded_file = SimpleUploadedFile(
            filename, file.read(), content_type='multipart/form-data')
        client = APIClient()
        client.force_authenticate(user=user)
        data = {
            'name': 'test',
            'description': 'test',
            'price': 1,
            'amount': 1,
            'image': uploaded_file,
            'owner': user.id
        }
        url = reverse('product-list')
        response = client.post(url, data, format='multipart')

        self.assertEqual(response.status_code, status.HTTP_201_CREATED)
        self.assertEqual(Product.objects.count(), 1)

        product = Product.objects.filter(name=data['name'])[0]
        self.assertEqual(product.name, data['name'])
        self.assertEqual(product.description, data['description'])
        self.assertEqual(product.price, data['price'])
        self.assertEqual(product.amount, data['amount'])
        self.assertEqual(product.image, data['image'])
        self.assertEqual(product.owner, data['owner'])


factories.py

import factory.fuzzy
import random
import datetime

from django.contrib.auth import get_user_model


class UserFactory(factory.django.DjangoModelFactory):
    class Meta:
        model = get_user_model()
        django_get_or_create = ('username',)

    email = factory.Sequence(lambda n: f'person{n}@example.com')
    username = factory.Faker('name')
    address = factory.Faker('address')
    icon = factory.Faker('image_url')
    message = factory.Faker('word')
    last_login_date = datetime.date.today()


models.py

from django.db import models


class Product(models.Model):
    """商品"""
    name = models.CharField(max_length=100)
    description = models.TextField(blank=True)
    price = models.PositiveIntegerField(default=0)
    amount = models.PositiveIntegerField(default=1)
    image = models.ImageField(upload_to='product')
    owner = models.ForeignKey('users.User', on_delete=models.CASCADE)
    created_at = models.DateTimeField(auto_now_add=True)

    def __str__(self):
        return self.name


views.py

from rest_framework import viewsets
from rest_framework import permissions, authentication

from .models import Product, Sale
from .serializers import ProductSerializer

class ProductViewSet(viewsets.ModelViewSet):
    """API endpoint"""
    queryset = Product.objects.all().order_by('-created_at')
    serializer_class = ProductSerializer
    permission_classes = [permissions.IsAuthenticated]


serializer.py

from rest_framework import serializers
from .models import Product


class ProductSerializer(serializers.ModelSerializer):
    class Meta:
        model = Product
        fields = ('id', 'name', 'description', 'price',
                  'amount', 'image', 'owner', 'created_at')

試したこと

client.post()で送るdataのimageフィールドを、初めは
data={'image': 'static/default_icon.jpg'}
のように直接書いていたのですが、同様に400エラーが出たので原因はこの部分かと思い上記のように修正しました。

このAPIが201を返すようにして、テストを通過させるにはどのようにすればいいですか?
よろしくお願いします。

参考にしたサイト

How to generate a file upload (test) request with Django REST Framework's APIRequestFactory?

公式ドキュメント

補足情報(FW/ツールのバージョンなど)

python==3.6.8
django==2.2.12
djangorestframework==3.11.0
pillow==7.0.0

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

check解決した方法

0

response.dataの中身を確認したところ、'image'フィールドに無効な値が入力されていることが分かったため、filenameを拡張子付きの文字列に修正しました。
これで上手くいきました。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 88.92%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る