Django - Сохранение сгенерированного PDF в модели с xhtml2pdf

Я пытаюсь сохранить сгенерированный PDF-файл с помощью приложения xhtml2pdf. Я хочу, чтобы он сохранил pdf в модели. PDF-файл создается идеально, и браузер загружает файл, но вместо загрузки я хотел бы сохранить его в модели, которую в моем models.py я сказал ему сохранить в папку в папке мультимедиа.

Models.py:

class purchase(models.Model):
    purchase_id = models.AutoField(primary_key=True)
    transaction_date = models.DateTimeField()
    method_payment = models.CharField(max_length = 20)
    cmr_name = models.CharField(max_length = 60)
    cmr_address = models.CharField(max_length = 90)
    cmr_postcode = models.CharField(max_length = 10)
    cmr_tel = models.CharField(max_length = 14)
    cmr_email = models.CharField(max_length = 40)
    cmr_pod = models.FileField(upload_to='customers_id')
    cmr_signature = JSignatureField()
    receipt = models.FileField(upload_to='receipts', null=True, blank=True)
    added_by = models.CharField(max_length = 30)
    item_brand = models.CharField(max_length = 50)
    product = models.CharField(max_length = 100)
    model = models.CharField(max_length = 100)
    serial = models.CharField(max_length = 50)
    item_brand2 = models.CharField(max_length = 50)
    product2 = models.CharField(max_length = 100)
    model2 = models.CharField(max_length = 100)
    serial2 = models.CharField(max_length = 50)
    item_brand3 = models.CharField(max_length = 50)
    product3 = models.CharField(max_length = 100)
    model3 = models.CharField(max_length = 100)
    serial3 = models.CharField(max_length = 50)
    def __str__(self):
        return '{}'.format(self.cmr_name)

Utils.py:

from io import BytesIO
from django.http import HttpResponse
from django.template.loader import get_template

from xhtml2pdf import pisa

def render_to_pdf(template_src, context_dict={}):
    template = get_template(template_src)
    html  = template.render(context_dict)
    result = BytesIO()
    pdf = pisa.pisaDocument(BytesIO(html.encode("ISO-8859-1")), result)
    if not pdf.err:
        return HttpResponse(result.getvalue(), content_type='application/pdf')
    return None

views.py:

@login_required(login_url='/accounts/login/')
def add_purchase(request):
    username = None
    if request.user.is_authenticated:
        username = request.user.username
    viewing_info = pwl.objects.get(pk='1')
    viewing_ot = opening_times.objects.get(pk='1')
    viewing_brands = watch_brands.objects.all()
    viewing_watches = watch.objects.values_list('watch_brand_name', flat=True).distinct()
    template = loader.get_template('pwl-access/add_purchase.html')
    if request.method == 'POST':
        form = add_purchase_form(request.POST, request.FILES)
        if form.is_valid():
            ## Form Data
            transaction_date = str(datetime.now().strftime('%Y-%m-%d %H:%M:%S'))
            added_by = request.user
            cmr_name = form.cleaned_data['cmr_name']
            cmr_address = form.cleaned_data['cmr_address']
            cmr_postcode = form.cleaned_data['cmr_postcode']
            cmr_tel = form.cleaned_data['cmr_tel']
            cmr_email = form.cleaned_data['cmr_email']
            purchase_price = form.cleaned_data['purchase_price']
            method_payment = form.cleaned_data['method_payment']
            item_brand = form.cleaned_data['item_brand']
            product = form.cleaned_data['product']
            model = form.cleaned_data['model']
            serial = form.cleaned_data['serial']
            price = form.cleaned_data['price']
            item_brand2 = form.cleaned_data['item_brand2']
            product2 = form.cleaned_data['product2']
            model2 = form.cleaned_data['model2']
            serial2 = form.cleaned_data['serial2']
            price2 = form.cleaned_data['price2']
            item_brand3 = form.cleaned_data['item_brand3']
            product3 = form.cleaned_data['product3']
            model3 = form.cleaned_data['model3']
            serial3 = form.cleaned_data['serial3']
            price3 = form.cleaned_data['price3']
            cmr_pod = request.FILES['cmr_pod'] if 'cmr_pod' in request.FILES else False
            cmr_signature = form.cleaned_data.get('cmr_signature')
            receipt = False
            if cmr_signature:
                signature_picture = draw_signature(cmr_signature, as_file=True)
            new_purchase = purchase(transaction_date=transaction_date,
                                    added_by=added_by,
                                    cmr_name=cmr_name,
                                    cmr_address=cmr_address,
                                    cmr_postcode=cmr_postcode,
                                    cmr_tel=cmr_tel,
                                    cmr_email=cmr_email,
                                    cmr_pod=cmr_pod,
                                    cmr_signature=signature_picture,
                                    receipt=receipt,
                                    method_payment=method_payment,
                                    item_brand=item_brand,
                                    product=product,
                                    model=model,
                                    serial=serial,
                                    item_brand2=item_brand2,
                                    product2=product2,
                                    model2=model2,
                                    serial2=serial2,
                                    item_brand3=item_brand3,
                                    product3=product3,
                                    model3=model3,
                                    serial3=serial3, )
            new_purchase.save()

            ##Receipt Creation
            cb_data = {
                'purchase_price': purchase_price,
                'price': price,
                'price2': price2,
                'price3': price3,
                'new_purchase': new_purchase,
                'cmr_pod': cmr_pod,
                'signature_picture': signature_picture,
            }
            template = get_template('pwl-access/email-templates/receipt_p_email.html')
            pdf = render_to_pdf('pwl-access/email-templates/receipt_p_pdf.html', cb_data)
            purch_id = new_purchase.pk
            if pdf:
                response = HttpResponse(pdf, content_type='application/pdf')
                filename = 'purchase_%s.pdf' % (purch_id)
                download = request.GET.get('download')
                content = "attachment; filename=%s " % (filename)
                response['Content-Disposition'] = content
                receipt_file = File(BytesIO(pdf.content))
                purchase.objects.filter(pk=purch_id).update(receipt=receipt_file)

            email_subject = 'Your Receipt from  - ' + transaction_date
            email_to = 'email'
            filled_email = template.render(cb_data)
            msg = EmailMultiAlternatives(email_subject, filled_email, '[email protected]',
                                         [email_to], )
            msg.content_subtype = "html"

            if email_subject:
                try:
                    msg.send()
                except BadHeaderError:
                    return HttpResponse('Invalid header found.')
                messages.success(request, 'Success! Purchase added and email has been sent to the customer ')
                return response
    else:
        form = add_purchase_form()

    context = {
        'viewing_info': viewing_info,
        'viewing_ot': viewing_ot,
        'viewing_brands': viewing_brands,
        'viewing_watches': viewing_watches,
        'form': form,
    }
    return HttpResponse(template.render(context, request))

Администратор и база данных показывают, что ничего не загружено, я не получаю ошибок. Я надеялся, что в файле views.py эти строки кода обновят базу данных с помощью файла pdf:

receipt_file = File(BytesIO(pdf.content))
purchase.objects.filter(pk=purch_id).update(receipt=receipt_file)

К сожалению, ничего не делает. Кто-нибудь может помочь? застрял в этом какое-то время. Я пробовал несколько других ответов на stackoverflow и других веб-сайтах.

Спасибо, любая помощь будет принята с благодарностью.

Edit1: я пытался вызвать save () вместо update, но это не имело значения:

                receipt_file = File(BytesIO(pdf.content))
                purch = purchase.objects.get(pk=purch_id)
                purch.receipt = receipt_file
                purch.save()

Edit2: я также пробовал несколько версий контента, и IOString (контент) ничего не работает

Edit3: Все еще нет ответа, какие-либо эксперты Django, которые могут знать, что происходит? Вроде такая простая вещь ..


person Akram Alkhatar    schedule 26.08.2020    source источник
comment
Почему бы просто не сгенерировать PDF-файл, как и когда кто-то захочет его увидеть? имеет больше смысла, потому что ваши объекты базы данных будут меньше, а эффективность - главное.   -  person markwalker_    schedule 26.08.2020
comment
В какой-то момент мы можем захотеть экспортировать файлы с помощью другой программы, используя предложенный вами способ, что означало бы необходимость заходить в каждую и загружать ее отдельно.   -  person Akram Alkhatar    schedule 26.08.2020
comment
Ответ на stackoverflow.com/questions/54040591/ может вам помочь. Это ImageField, а не FileField, но эта часть ответа кажется полезной: метод QuerySet.update () не вызывает save () в модели, поэтому обычный механизм, помещающий изображение в хранилище, не выполняется. Это может быть то, что происходит и с вами - надеюсь, ответ на этот вопрос поможет!   -  person bouteillebleu    schedule 26.08.2020
comment
Спасибо за предложение, функция save () ничего не изменила: Receipt_file = File (BytesIO (pdf.content)) purch = Purchase.objects.get (pk = purch_id) purch.receipt = Receipt_file purch.save ()   -  person Akram Alkhatar    schedule 26.08.2020


Ответы (1)


В итоге мне удалось это сделать с помощью следующего кода:

                receipt_file = BytesIO(pdf.content)
                purch_upd = purchase.objects.get(pk=purch_id)
                purch_upd.receipt = File(receipt_file, filename)
                purch_upd.save()

так просто, не могу поверить, что застрял на этом так долго.

person Akram Alkhatar    schedule 10.11.2020
comment
Какая функция - это файл? Куда вы его ввозите? - person Darwin; 25.01.2021
comment
@ Дарвин из django.core.files import File - person Ankit Tiwari; 05.02.2021
comment
@Akram Alkhatar Спасибо, что задали этот вопрос, я узнал кое-что новое из вашего кода (-: - person Ankit Tiwari; 05.02.2021
comment
Пожалуйста, извините за такой поздний ответ! - person Akram Alkhatar; 21.03.2021