TypeError : Поле 'id' ожидало число, но получило '<django.db.models.query_utils.DeferredAttribute object at 0x00000000044871C0>'
Доброго времени суток. Я начинающий Django разработчик. Работая над проектом, я столкнулся с проблемой, а старших товарищей у меня нет. Не могли бы вы мне с ней помочь?
Я разрабатываю небольшой магазин и при добавлении товара в корзину получаю эту ошибку. Я уже все перепробовал и просто в отчаянии
Поле 'id' ожидало число, но получило "<django.db.models.query_utils.DeferredAttribute object at 0x00000000044871C0>'.
models.py
# Категория продукта
class Category(models.Model):
name = models.CharField('Категория', max_length=100)
url = models.SlugField(max_length=100, unique=True, db_index=True)
def __str__(self):
return f'{self.name}'
def save(self, *args, **kwargs):
if not self.url:
self.url = slugify(self.name)
super(Category, self).save(*args, **kwargs)
def get_absolute_url(self):
return reverse('product_list_by_category', args=[self.url])
class Meta:
verbose_name = "Категория"
verbose_name_plural = "Категории"
# Папка продуктов для каждой организации
def products_image_path(instance, filename):
return f'organisations_{instance.organisations.name}/products/{filename}'
# Продукт
class Products(models.Model):
name = models.TextField(max_length=100, blank=False)
slug = models.SlugField(max_length=200, db_index=True)
image = models.ImageField(upload_to=products_image_path, blank=True)
number = models.PositiveIntegerField('Количество')
price = models.DecimalField('Цена', max_digits=10, decimal_places=2)
category = models.ForeignKey(Category, verbose_name='Категория', blank=False, on_delete=models.SET_NULL, null=True)
organisations = models.ForeignKey(Organisations, related_name="org_info", on_delete=models.CASCADE)
def __str__(self):
return f'{self.name}'
def save(self, *args, **kwargs):
if not self.slug:
self.slug = slugify(self.name)
super(Products, self).save(*args, **kwargs)
def get_absolute_url(self):
return reverse('product_detail', args=[self.id, self.slug])
class Meta:
verbose_name = "Продукт"
verbose_name_plural = "Продукты"
ordering = ('name',)
index_together = (('id', 'slug'),)
# Покупатель
class Customer(models.Model):
card = models.CharField("Номер карты",max_length=19)
telephoneRegex = RegexValidator(regex = r"^\+?1?\d{8,15}$")
telephone = models.CharField("Номер телефона", validators = [telephoneRegex], max_length = 16, unique = True)
email = models.EmailField("Электронная почта", max_length=50)
name = models.CharField("Имя", max_length=100)
discount = models.BigIntegerField('Скидка')
score = models.BigIntegerField('Счет')
organisations = models.ForeignKey(Organisations, on_delete=models.CASCADE)
def __str__(self):
return f'{self.name}'
class Meta:
verbose_name = "Клиент"
verbose_name_plural = "Клиенты"
Views.py
from django.shortcuts import render, redirect, get_object_or_404
from django.views.decorators.http import require_POST
from Warehouse.models import Products
from .cart import Cart
from .forms import CartAddProductForm
@require_POST
def cart_add(request, product_id):
cart = Cart(request)
product = get_object_or_404(Products, id=product_id)
form = CartAddProductForm(request.POST)
if form.is_valid():
cd = form.cleaned_data
cart.add(product=product, quantity=cd['quantity'],
update_quantity=cd['update'])
return redirect('cart_detail')
def cart_remove(request, product_id):
cart = Cart(request)
product = get_object_or_404(Products, id=product_id)
cart.remove(product)
return redirect('cart_detail')
def cart_detail(request):
cart = Cart(request)
for item in cart:
item['update_quantity_form'] = CartAddProductForm(
initial={
'quantity': item['quantity'],
'update': True
})
return render(request, 'cart/detail.html', {'cart': cart})
forms.py
from django import forms
PRODUCT_QUANTITY_CHOICES = [(i, str(i)) for i in range(1, 21)]
#Добавление товара в корзину
class CartAddProductForm(forms.Form):
quantity = forms.TypedChoiceField(choices=PRODUCT_QUANTITY_CHOICES, coerce=int)
update = forms.BooleanField(required=False, initial=False, widget=forms.HiddenInput)
cart.py
import OnliCas.settings
from decimal import Decimal
from Warehouse.models import Products
class Cart(object):
def __init__(self, request):
# Инициализация корзины пользователя
self.session = request.session
cart = self.session.get(OnliCas.settings.CART_SESSION_ID)
if not cart:
# Сохраняем корзину пользователя в сессию
cart = self.session[OnliCas.settings.CART_SESSION_ID] = {}
self.cart = cart
# Добавление товар в корзину пользователя
# или обновление количества товаров
def add(self, product, quantity=1, update_quantity=False):
product_id = str(product.id)
if product_id not in self.cart:
self.cart[product_id] = {'quantity': 0,
'price': str(product.price)}
if update_quantity:
self.cart[product_id]['quantity'] = quantity
else:
self.cart[product_id]['quantity'] += quantity
self.save()
# Сохранение данных в сессию
def save(self):
self.session[OnliCas.settings.CART_SESSION_ID] = self.cart
# Указываем, что сессия изменена
self.session.modified = True
# Удаление товара из корзины
def remove(self, product):
product_id = str(product.id)
if product_id in self.cart:
del self.cart[product_id]
self.save()
# Итерация по товарам
def __iter__(self):
product_ids = self.cart.keys()
products = Products.objects.filter(id__in=product_ids)
for product in products:
self.cart[str(product.id)]['product'] = product
for item in self.cart.values():
item['price'] = Decimal(item['price'])
item['total_price'] = item['price'] * item['quantity']
yield item
# Количество товаров
def __len__(self):
return sum(item['quantity'] for item in self.cart.values())
def get_total_price(self):
return sum(Decimal(item['price'])*item['quantity'] for item in self.cart.values())
def clear(self):
del self.session[OnliCas.settings.CART_SESSION_ID]
self.session.modified = True
settings.py
# session settings
CART_SESSION_ID = 'cart'