Путь к промежуточной модели для многих ко многим в Django Rest Framework - PullRequest
1 голос
/ 21 марта 2019
  • Я хотел бы узнать ваше мнение, правильный ли это подход?
  • Как лучше сериализовать промежуточную модель m2m?
  • Я хотел бы знать, как объединить сериализаторы Display и Create ..
  • models.py
class Material(models.Model):
    name = models.CharField(max_length=120)   


class Product(models.Model):
    name = models.CharField(max_length=120)
    materials = models.ManyToManyField(Material, through='MaterialProduct')


class MaterialProduct(models.Model):
    material = models.ForeignKey(Material, on_delete=models.CASCADE)
    product = models.ForeignKey(Product, on_delete=models.CASCADE)
    rate = models.FloatField(default=100)
  • views.py
class Products(APIView):
    def get(self, request, format=None):
        products = Product.objects.all()
        serializer = ProductDisplaySerializer(products, many=True) # Display
        return Response(serializer.data)

    def post(self, request, format=None):
        serializer = ProductCreateSerializer(data=request.data) # Create
        if serializer.is_valid():
            serializer.save()
            return Response(serializer.data, status=status.HTTP_201_CREATED)
        return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)

  • serializers.py

class MaterialSerializer(serializers.ModelSerializer):
    class Meta:
        model = Material
        fields = '__all__'


class ProductMaterialRateSerializer(serializers.ModelSerializer):
    material = MaterialSerializer(read_only=True)
    material_id = serializers.PrimaryKeyRelatedField(
        write_only=True, source='material', queryset=Material.objects.all())

    class Meta:
        model = MaterialProduct  # attention!!!
        fields = ('material', 'material_id', 'rate')


class ProductCreateSerializer(serializers.ModelSerializer):
    '''To create a product with existed material and a material rate(extra field) '''
    materials = ProductMaterialRateSerializer(many=True)

    class Meta:
        model = Product
        fields = ('id', 'name', 'materials')

    def create(self, validated_data):

        materials_data = validated_data.pop('materials')
        product = Product.objects.create(**validated_data)
        for material_data in materials_data:
            MaterialProduct.objects.create(
                product=product,
                material=material_data.get('material'),
                rate=material_data.get('rate'))
        return product
  • Отображаемые данные:
 {
        "id": 29,
        "name": "product 4",
        "materials": [
            {
                "material_id": 3,
                "rate": 30
            },
            {
                "material_id": 2,
                "rate": 70
            }
        ]
    }

Создание данных для отправки:

{
    "name" : "product 4",
    "materials" : [
        {
            "material_id":3,
            "rate" : 30
        }   
        ,{
            "material_id":2,
            "rate" : 70
        }

    ]
}

Возвращенные данные после создания:

  • Примечание. Сохраняетданные правильно, но не отображаются, как показано ниже!
{
    "id": 29,
    "name": "product 4",
    "materials": [
        {},
        {}
    ]
}

1 Ответ

1 голос
/ 21 марта 2019

Вы можете использовать RetrieveAPIView для отображения и CreateAPIView для создания, которое является встроенными функциями и может упростить задачу, а другим разработчикам будет проще работать с вашим кодом, потому что все, с чем им нужно ознакомиться, это то, как работает DRF. как вы написали код. По сути, лучше следовать шаблонам, с которыми знакомы другие разработчики.

И я не рекомендую комбинировать создание и отображение сериализаторов, если они не очень просты. если они немного усложняются, может быть сложно отлаживать или вносить изменения, особенно если кто-то другой отлаживает ваш код!

А для сериализации m2m materials = ProductMaterialRateSerializer(many=True) - это правильный путь, и для создания вы можете использовать встроенные функции, но если это сложно и требует некоторых вычислений, то я рекомендую переопределить метод save и update в вашем сериализаторе создания.

и для возвращенных данных:

Вы можете переопределить контекст следующим образом:

def post(self, request, format=None):
    serializer = ProductCreateSerializer(data=request.data) # Create
    if serializer.is_valid():

        # Get the saved object
        saved_obj = serializer.save()

        # Serialize the saved object with the preferred serializer.
        response_data = ProductDisplaySerializer(saved_obj).data

        return Response(response_data, status=status.HTTP_201_CREATED)
    return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...