如何通过 Django 中的自定义管理操作获取关联模型?

How to get an associated model via a custom admin action in Django?

本题分开提问和回答。

我有一个 Report 和一个 ReportTemplate

+----+----------+---------------+-------------+
| id |  title   |     data      | template_id |
+----+----------+---------------+-------------+
|  1 | report 1 | {data: [...]} |           1 |
+----+----------+---------------+-------------+

reports table

+----+-----------+---------------+------------+
| id |   title   |    markup     |    css     |
+----+-----------+---------------+------------+
|  1 | template1 | <doctype!>... | body {.... |
+----+-----------+---------------+------------+

templates table

报表属于报表模板。一个 ReportTemplate 有很多 Report。

我在 admin.py 中有一个名为 print_as_pdf

的自定义管理操作
class ReportAdmin(admin.ModelAdmin):
    fields = ['commodity', 
    'date',
    'trade_period',
    'quantity_cutoff',
    'data',
    'template',
    'title']

    actions = ['print_as_pdf']

    def print_as_pdf(self, request, queryset):
        return
    print_as_pdf.short_description = 'Generate as pdf'

这些是模型:

class ReportTemplate(models.Model):
    title = models.CharField(max_length=50)
    markup = models.TextField(default = 'markup here...')
    styles = models.TextField(default = 'styles here...')

    # __unicode__ on Python 2
    # __str__ on Python 3
    def __unicode__(self):
        return self.title

class Report(models.Model):
    title = models.CharField(max_length=50)
    commodity = models.CharField(max_length=10)
    date = models.DateTimeField('date traded')
    trade_period = models.CharField(max_length=10, default='open')
    quantity_cutoff = models.IntegerField(default=0)
    printed = models.BooleanField(default=0)
    datetime_email_sent = models.DateTimeField('date email sent', blank=True, null=True)
    data = models.TextField(default = 'data here...')
    template = models.ForeignKey(ReportTemplate)

我想做的是:

  1. 检索关联的 ReportTemplate 及其 markup 字段值
  2. 将Report的data字段值通过jinja2 markup写的markup值放在1
  3. 使用 weasyprint 并将 2 中的数据填充标记打印为 pdf

我卡在第 1 步了。

给定参数 selfrequestqueryset,如何检索关联的 ReportTemplate 及其 markup 字段值?

更新 1:

我试过这个来测试给出的答案之一。

import logging

logger = logging.getLogger(__name__)

# .... code here ...

def print_as_pdf(self, request, queryset):
        for report in queryset:
            markup = report.template.markup
            logger.debug(markup)
        return

更新 2:

# Logging
LOGGING = {
    'version': 1,
    'disable_existing_loggers': False,
    'handlers': {
        'file': {
            'level': 'DEBUG',
            'class': 'logging.FileHandler',
            'filename': '/var/virtual/WebApps/virtualenvs/WeasyPrintProject/weasyprint_site/debug.log',
        },
    },
    'loggers': {
        'reports.admin': {
            'handlers': ['file'],
            'level': 'DEBUG',
            'propagate': True,
        },
    },
}

写在我的 settings.py

生成了一个debug.log

然而,debug.log 的内容是空的

更新 3:

需要明确更改

logger = logging.getLogger(__name__)

logger = logging.getLogger('reports.admin')

只需获取 Report 模型的 template 字段:

def print_as_pdf(self, request, queryset):
    for report in queryset:
        markup = report.template.markup
        ...
print_as_pdf.short_description = 'Generate as pdf'

UPDATE:为了use the logger你应该在源文件的开头添加这两行:

import logging

logger = logging.getLogger(__name__)