如何通过 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)
我想做的是:
- 检索关联的 ReportTemplate 及其
markup
字段值
- 将Report的
data
字段值通过jinja2 markup写的markup
值放在1
- 使用 weasyprint 并将 2 中的数据填充标记打印为 pdf
我卡在第 1 步了。
给定参数 self
、request
、queryset
,如何检索关联的 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__)
我有一个 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)
我想做的是:
- 检索关联的 ReportTemplate 及其
markup
字段值 - 将Report的
data
字段值通过jinja2 markup写的markup
值放在1 - 使用 weasyprint 并将 2 中的数据填充标记打印为 pdf
我卡在第 1 步了。
给定参数 self
、request
、queryset
,如何检索关联的 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__)