Django поиск: запросить несколько таблиц, присоединиться к ним и вернуть набор запросов - PullRequest
1 голос
/ 18 марта 2020

Я строю поиск для моего приложения для отслеживания цен на товары, и есть 2 модели продуктов и сканирований

class products(models.Model):
    enter code hereproduct_id = models.IntegerField(primary_key=True)
    product_name = models.CharField(max_length=100)
    product_link = models.CharField(max_length=255)
    image_link = models.CharField(max_length=100)
    store_name = models.CharField(max_length=100)

    class Meta:
        verbose_name_plural = "products"

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

class scans(models.Model):
    scan_id = models.IntegerField(primary_key=True)
    prod_id = models.IntegerField(models.ForeignKey("scrape.Model", on_delete=models.CASCADE))
    price = models.IntegerField()
    scanned_time = models.DateTimeField(default=timezone.now)

    class Meta:
      verbose_name_plural = "scans"

    def __repr__(self):
        return str(self.prod_id)

при поиске необходимо ввести имя (строку) и запрос для магазина имя в той же таблице и запрос цены, которая находится в другой таблице, и возвращает набор запросов с именем продукта, именем магазина и ценой.

Я пробовал это

class searchResults(ListView):
model = 'products', 'scans'
template_name = 'result.html'

# let query = GET request from form (in home.html)
def get_queryset(self):
    queryset = []
    rawquery = self.request.GET.get('q')
    queries = rawquery.split(" ")
    for q in queries:
        prodresults = products.objects.filter(
            Q(product_name__icontains=q)
        ).distinct()

        for prodresult in prodresults:
            prod_id = products.objects.get(product_name=prodresult).product_id
            prod_price = scans.objects.get(prod_id=prod_id).price

        for result in prodresults:
            #joins results toegther
            queryset.append(result)
            queryset.append(prod_price)

    return list(set(queryset))

Код сломался в

prod_price = scans.objects.get(prod_id=prod_id).price

с scans matching query does not exist.

1 Ответ

1 голос
/ 18 марта 2020

Ваши модели на самом деле не определены правильно с точки зрения именования, кодирования и Django соглашений. Вы должны изменить их на это:

class Product(models.Model):  # Capital P, singular
    id = models.IntegerField(primary_key=True)  # id, not product_id
    name = models.CharField(max_length=100)  # name, not product_name
    link = models.CharField(max_length=255)
    image_link = models.CharField(max_length=100)
    store_name = models.CharField(max_length=100)

    # ...

class Scan(models.Model):
    id = models.IntegerField(primary_key=True)
    product = models.ForeignKey("Product", on_delete=models.CASCADE, related_name="scans")  # FK to Product
    price = models.IntegerField()
    scanned_time = models.DateTimeField(default=timezone.now)

    # ...

Затем, когда у вас есть набор запросов продуктов:

results = Product.objects.filter(product_name__icontains=q).distinct()

, вы можете через них l oop и получить соответствующие сканы:

for product in results:
    scans = Scan.objects.filter(product=product)  # You can have more than one Scan for the same product, since this is a one-to-many relationship!

Лучше, вы можете сделать все это одним запросом:

scans = Scan.objects.filter(product__in=Product.objects.filter(
    product_name__icontains=q))
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...