Получение связанных внешних ключей модели (в дополнение к ManyToMany) с помощью prefetch_related(model_SET).all()

С помощью prefetch_related я получаю список запасов различных складов следующим образом :

Модель продукта

[ничего особенного]

Модельное хранилище

[ничего особенного]

Модель SstStock

class SstStock(models.Model):
    warehouse = models.ForeignKey(Warehouse, on_delete=models.CASCADE)
    product = models.ManyToManyField(Produit)
    qty = models.IntegerField()
    last_update = models.DateTimeField(default=timezone.now)

views.py

class ProductListView(LoginRequiredMixin, ListView):
    queryset = Product.objects.prefetch_related('sststock_set').all()
    context_object_name = "produits"
    paginate_by = 10
    template_name = 'products/produits.html'

    def get_context_data(self, *args, **kwargs):
        context = super(ProductListView, self).get_context_data(
            *args, **kwargs)
        context['title'] = 'Produits'
        return context

Шаблон

{% for produit in produits %}
    {{ produit.sku}}<br>
    {% for sst in produit.sststock_set.all %}
        <span>{{ sst.warehouse.code }} - {{sst.qty}} - {{sst.qty}}</span>
    {% endfor %}
{% endfor %}

Но когда я получаю информацию, связанную со складом, например sst.warehouse.code в шаблоне, количество запросов взрывается. (уже 36 запросов с пагинацией только 10 товаров)

Есть ли способ добавить склад в prefetch_related в представлении ?

Вы можете использовать Prefetch объект [Django-doc]:

class ProductListView(LoginRequiredMixin, ListView):
    queryset = Product.objects.prefetch_related(
        Prefetch('sststock_set', SstStock.objects.select_related('warehouse'))
    ))
    # …

Примечание: [Django-doc]DateTimeField [Django-doc]. имеет auto_now=… параметр [Django-doc] для работы с временными метками. Это позволит автоматически присваивать текущее время при обновлении объекта, и пометит его как нередактируемый (editable=False), так что что оно не будет отображаться в ModelForm по умолчанию.

Ну, поскольку я не мог найти способ продолжить работу с сериализатором для получения моего sststock_set, единственным решением на тот момент было перестроить json данных в представлении.

data = list()
        for product in products:
            item = {}
            item['sku'] = product.sku
            item['pk'] = product.pk
            item['famille'] = product.famille.nom
            item['etat'] = product.etat.etat
            # print(product.nom)
            item['mageid'] = product.mageid
            item['adresse'] = product.adresse
            item['fournisseur'] = product.fournisseur.nom
            item['mini'] = product.mini
            item['pua'] = float(product.pua)
            item['pta'] = float(product.pta)
            item['nom'] = product.nom
            item['puv'] = float(product.puv)
            item['ptv'] = float(product.ptv)
            item['cau'] = float(product.cau)
            item['cau_cli'] = float(product.cau_cli)
            item['maxsst2'] = product.maxsst2

            item['sststock'] = list()
            for sst in product.sststock_set.all():
                item['sststock'].append(
                    {'sst_code': sst.warehouse.code, 'sst_id': sst.warehouse.pk, 'qty': sst.qty})
            data.append(item)
        data = json.dumps(data)
        
        return HttpResponse(data, content_type="application/json")
Вернуться на верх