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

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

新規登録して質問してみよう
ただいま回答率
85.35%
Django

DjangoはPythonで書かれた、オープンソースウェブアプリケーションのフレームワークです。複雑なデータベースを扱うウェブサイトを開発する際に必要な労力を減らす為にデザインされました。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

Q&A

解決済

1回答

2526閲覧

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

shun173

総合スコア4

Django

DjangoはPythonで書かれた、オープンソースウェブアプリケーションのフレームワークです。複雑なデータベースを扱うウェブサイトを開発する際に必要な労力を減らす為にデザインされました。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

0グッド

0クリップ

投稿2020/07/10 12:38

編集2020/07/12 03:36

前提・実現したいこと

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

python

1from django.core.files import File 2from django.core.files.uploadedfile import SimpleUploadedFile 3from django.test import TestCase, Client 4from django.test.client import encode_multipart 5from django.urls import reverse 6from django.utils.timezone import localtime 7from faker import Faker 8from rest_framework import status 9from rest_framework.test import APIRequestFactory, force_authenticate, APIClient, APITestCase 10 11from ..models import Product 12from ..views import ProductViewSet 13from .factories import ProductFactory, UserFactory 14 15 16class ProductApiTest(APITestCase): 17 def test_post(self): 18 user = UserFactory() 19 view = ProductViewSet.as_view({'post': 'create'}) 20 filename = 'test_img' 21 file = File(open('static/default_icon.jpg', 'rb')) 22 uploaded_file = SimpleUploadedFile( 23 filename, file.read(), content_type='multipart/form-data') 24 client = APIClient() 25 client.force_authenticate(user=user) 26 data = { 27 'name': 'test', 28 'description': 'test', 29 'price': 1, 30 'amount': 1, 31 'image': uploaded_file, 32 'owner': user.id 33 } 34 url = reverse('product-list') 35 response = client.post(url, data, format='multipart') 36 37 self.assertEqual(response.status_code, status.HTTP_201_CREATED) 38 self.assertEqual(Product.objects.count(), 1) 39 40 product = Product.objects.filter(name=data['name'])[0] 41 self.assertEqual(product.name, data['name']) 42 self.assertEqual(product.description, data['description']) 43 self.assertEqual(product.price, data['price']) 44 self.assertEqual(product.amount, data['amount']) 45 self.assertEqual(product.image, data['image']) 46 self.assertEqual(product.owner, data['owner'])

factories.py

python

1import factory.fuzzy 2import random 3import datetime 4 5from django.contrib.auth import get_user_model 6 7 8class UserFactory(factory.django.DjangoModelFactory): 9 class Meta: 10 model = get_user_model() 11 django_get_or_create = ('username',) 12 13 email = factory.Sequence(lambda n: f'person{n}@example.com') 14 username = factory.Faker('name') 15 address = factory.Faker('address') 16 icon = factory.Faker('image_url') 17 message = factory.Faker('word') 18 last_login_date = datetime.date.today() 19

models.py

python

1from django.db import models 2 3 4class Product(models.Model): 5 """商品""" 6 name = models.CharField(max_length=100) 7 description = models.TextField(blank=True) 8 price = models.PositiveIntegerField(default=0) 9 amount = models.PositiveIntegerField(default=1) 10 image = models.ImageField(upload_to='product') 11 owner = models.ForeignKey('users.User', on_delete=models.CASCADE) 12 created_at = models.DateTimeField(auto_now_add=True) 13 14 def __str__(self): 15 return self.name

views.py

python

1from rest_framework import viewsets 2from rest_framework import permissions, authentication 3 4from .models import Product, Sale 5from .serializers import ProductSerializer 6 7class ProductViewSet(viewsets.ModelViewSet): 8 """API endpoint""" 9 queryset = Product.objects.all().order_by('-created_at') 10 serializer_class = ProductSerializer 11 permission_classes = [permissions.IsAuthenticated] 12

serializer.py

python

1from rest_framework import serializers 2from .models import Product 3 4 5class ProductSerializer(serializers.ModelSerializer): 6 class Meta: 7 model = Product 8 fields = ('id', 'name', 'description', 'price', 9 'amount', 'image', 'owner', 'created_at') 10

試したこと

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

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

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

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

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

自己解決

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

投稿2020/07/12 03:41

shun173

総合スコア4

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.35%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問