繁体   English   中英

在 Django REST Framework 中测试图片上传

[英]Testing image uploads in Django REST Framework

我正在尝试向 Django 项目中的一个应用程序添加一些测试。 此应用程序包含Image model:

from django.db import models
from django.utils.translation import gettext_lazy as _

from cecommerce.image_mappings import ImageAgnosticMapping
from cecoresizer.fields import ResizableImageField


class Image(models.Model):
    file = ResizableImageField(
        _("Image"),
        max_length=300,
        image_config=ImageAgnosticMapping,
    )

    def __str__(self):
        return str(self.file)

然后使用以下序列化程序对其进行序列化:

from django.http import Http404

from rest_framework import serializers

from cecotec_apps.landings.models import ProductLanding
from cecotec_apps.partner.models import Home

from images.models import Image


class ImageSerializer(serializers.ModelSerializer):
    file = serializers.ImageField()

    def to_representation(self, instance):
        return {"file": str(instance)}

    class Meta:
        model = Image
        exclude = ("id",)

我正在尝试通过此测试来测试Image实例的创建(在使用 Imsonia 请求 API 时有效):

import tempfile

from PIL import Image as ImageFile

from django.test import tag

from model_bakery import baker

from rest_framework.test import APITestCase
from rest_framework.authtoken.models import Token
from rest_framework.reverse import reverse


from cecotec_apps.landings.models import ProductLanding
from cecotec_apps.partner.models import Home

from images.api.serializers import ImageSerializer
from images.models import Image

from tests.oscar.decorator.decorator_all_methods import global_test_decorator

from user.models import User


@tag("e2e", "image")
@global_test_decorator()
class ImageTestCase(APITestCase):
    API_VERSION = "v1"
    IMAGES_QUANTITY = 20
    HTTP_HOST = "localhost:8000"

    @classmethod
    def _populate(cls):
        cls.token = baker.make(Token)
        baker.make_recipe("images.image_recipe", _quantity=cls.IMAGES_QUANTITY)

    @classmethod
    def _generate_image_file(cls):
        with tempfile.NamedTemporaryFile(suffix="jpg") as tmp_file:
            image = ImageFile.new("RGB", size=(100, 100))
            image.save(tmp_file, "jpeg")
            tmp_file.seek(0)

            return tmp_file

    @classmethod
    def _depopulate(cls):
        Token.objects.all().delete()
        User.objects.all().delete()
        ProductLanding.objects.all().delete()
        Home.objects.all().delete()
        Image.objects.all().delete()

    def setUp(self, *args):
        self._depopulate()
        self._populate()

    def tearDown(self, *args):
        self._depopulate()

    @tag("create", "authenticated")
    def test_create_authenticated(self, *args):
        self.client.force_authenticate(user=self.token.user, token=self.token)

        data = self._prepare_create_data(**{"file": self._generate_image_file()})

        response = self._call_create(data)

        self._assert_create(response)
        self.assertRegex(response.data.get("file"), "^images\/.+$")

    def _call_create(self, data):
        return self.client.post(
            path=reverse(f"image-list"),
            data=data,
            # content_type=MULTIPART_CONTENT,
            format="multipart",
            HTTP_HOST=self.HTTP_HOST,
        )

    def _assert_create(self, response, expected_status_code=201):
        self.assertEqual(response.status_code, expected_status_code)

        if expected_status_code == 201:
            response_data = response.data

            self.assertEqual(Image.objects.count(), self.IMAGES_QUANTITY + 1)
            self._assert_response_body(response_data)
        else:
            self.assertEqual(Image.objects.count(), self.IMAGES_QUANTITY)

    def _assert_response_body(self, response_body):
        self.assertIn("file", response_body)

但是,运行它时返回的响应代码是400 Bad Request而不是201 Created 响应体是这样的:

{
    "file": [
        ErrorDetail(string="The submitted data was not a file. Check the encoding type on the form.",
        code="invalid")
    ]
}

我也试过使用已经创建的文件进行测试,但错误是一样的。 我在网上看到很多帖子,但没有一个能帮助我解决这个问题。

我会很感激一些帮助。 提前致谢。

原因

我认为问题来自_generate_image_file方法。

@classmethod
def _generate_image_file(cls):
    with tempfile.NamedTemporaryFile(suffix="jpg") as tmp_file:
        image = ImageFile.new("RGB", size=(100, 100))
        image.save(tmp_file, "jpeg")
        tmp_file.seek(0)

        return tmp_file

由于tempfile.NamedTemporaryFile是使用with语句创建的,因此当您返回它并在另一种方法中使用它时,该文件已经关闭并删除。

解决方案

您应该通过使用yied装饰方法将_generate_image_file更改为上下文管理器,并使用contextlib.contextmanager语句而不是 return。

from contextlib import contextmanager

@contextmanager
@classmethod
def _generate_image_file(cls):
    with tempfile.NamedTemporaryFile(suffix="jpg") as tmp_file:
        image = ImageFile.new("RGB", size=(100, 100))
        image.save(tmp_file, "jpeg")
        tmp_file.seek(0)

        yied tmp_file

并在test_create_authenticated方法上,将其更改为以下内容

@tag("create", "authenticated")
def test_create_authenticated(self, *args):
    ...
    with self._generate_image_file() as image_file:
        data = self._prepare_create_data(**{"file": image_file})
        response = self._call_create(data)
    ...

现在图像文件将在with块退出后被删除。

我建议使用这个:

from django.core.files.uploadedfile import SimpleUploadedFile

with open("/some/path/to/image.jpg", "rb") as image:
        image = SimpleUploadedFile("image.jpg", image.read(), content_type="image/jpg")
        data = self._prepare_create_data(**{"file": image})
...

发生的情况是您的序列化程序正在验证正在发送的文件无效。 确保在您的视图中有parser_classes = (parsers.MultiPartParser,) ,在您的models.ImageField中也有 models.ImageField。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM