Django(django_tables2)ValueError:预期的表或查询集,而不是str + for循环问题

时间:2018-10-18 13:04:32

标签: django django-models django-views django-tables2

我有一个受Mozilla的Django教程(https://developer.mozilla.org/en-US/docs/Learn/Server-side/Django/Tutorial_local_library_website)影响很大的Django项目。我已经将books修改为samples,并且每个samples都隶属于研究员/ pi(又名author)。我创建了一个名为Variant的单独模型,它是Sample模型的外键。由于此字段将为每个样本提供多个变体,因此在查看样本的详细视图时,我想将变体(特定于单个样本)显示为渲染表。我尚未成功实现这一目标,我认为我只是不太正确地获得了正确的视图命名法(以及html文件中的语言)。我的问题在这篇文章的结尾。

这是我的代码。

samples / models.py

class Variant(models.Model):
   gene = models.CharField('Gene', max_length=100, blank=True, default='')
   variant = models.CharField('Variant', max_length=50, blank=True, default='')
   call = models.CharField('Call', max_length=100, blank=True, default='')
   position = models.IntegerField('Position', blank=True, null=True) 

   def __str__(self):
      return f'{self.gene}, {self.position}, {self.variant}, {self.call}'

class Sample(models.Model):
   sample_name = models.CharField('Sample', max_length=16)
   pi = models.ForeignKey(PI, on_delete=models.SET_NULL, null=True)
   sample_variant = models.ForeignKey(Variant, on_delete=models.SET_NULL, null=True) 
     .
     .
     .
   def __str__(self):
      return self.sample_name

   def get_absolute_url(self):
      return reverse('sample-detail', args=[str(self.id)])

tables.py

class SampleTable(tables.Table):
   sample_name = tables.LinkColumn('sample-detail', args=[A('pk')])
   pi = tables.LinkColumn('sample-detail', args=[A('pk')])

   class Meta:
      model = Sample
#----------------------------------------------------------------------------

#----------------------------------------------------------------------------
class PISampleTable(tables.Table):
   sample_name = tables.LinkColumn('sample-detail', args=[A('pk')])

   class Meta:
      model = Sample
#----------------------------------------------------------------------------

...
#----------------------------------------------------------------------------
class VariantTable(tables.Table):
   gene = tables.LinkColumn('sample-detail', args=[A('pk')])

   class Meta:
      model = Variant
#----------------------------------------------------------------------------

我不确定像我一样设置基因(变体场)是否具有结肠功能。

samples / views.py

class SampleListView(generic.ListView):
   model = Sample
   paginate_by = 100

@login_required
def sam(request):
   table = SampleTable(Sample.objects.all())
   RequestConfig(request).configure(table)
   return render(request, 'samples/sample_list.html', {'sam': table}) 
#-------------------------------------------------------------------------------------------------

#-------------------------------------------------------------------------------------------------
class SampleDetailView(generic.DetailView):
    model = Sample
#-------------------------------------------------------------------------------------------------

    ...
#-------------------------------------------------------------------------------------------------
class VariantListView(generic.ListView):
   model = Variant
   paginate_by = 100

@login_required
def var(request):
   #table = VariantTable(Variant.objects.filter(pi=pk))
   table = VariantTable(Variant.objects.all())
   RequestConfig(request).configure(table)
   return render(request, 'samples/sample_detail.html', {'var': table}) 
#-------------------------------------------------------------------------------------------------

samples / urls.py

urlpatterns = [ 
   path('', views.index, name='index'),
   path('samples/', views.sam, name='sam'),
   path('sample/<int:pk>', views.SampleDetailView.as_view(), name='sample-detail'),
   path('pi/', views.pi_table, name='pi_table'),
   path('pi/<int:pk>', views.pi_view, name='pi-detail'),
   path('samples/new', views.pi_new, name='pi_new'),
   path('samples/new_sample', views.sample_new, name='sample_new'),
   #path('samples/', views.var, name='var'),
   #path('samples/<int:pk>', views.var, name='var'),
   path('samples/', views.VariantListView.as_view(), name='var'),
]

samples / templates / samples / sample_detail.html

 {% extends "base_generic.html" %}
{% load render_table from django_tables2 %}

{% block content %}
   <h1>Sample: {{ sample.sample_name }}</h1>

   <p><strong>PI:</strong> <a href="{% url 'pi-detail' sample.pi.pk %}">{{ sample.pi }}</a></p>
   ...
   <h3>Variant Results</h3>
   <p><strong>(gene, position, variant, call):</strong> {{ sample.sample_variant }}</p> <!-- this works, but only last item in list-->


  <p><strong>Variant:</strong> {% for variant in sample.sample_variant.all %} {{ variant }}{% if not forloop.last %}, {% endif %}{% endfor %}</p>

  {% if sample.sample_variant %}
  <ul>

   {% for variant in variant_list %}
     <li> <a href="{{ variant.get_absolute_url }}">{{ variant.gene }}</a> ({{variant.call}})</li>
   {% endfor %}
  </ul>
  {% else %}
    <p>There are no variants for this sample.</p>                                                                                                                                    

  {% endif %}       

{% render_table var %}

{% endblock %} 

要进行测试,我有一个示例,其中添加了两个变体-但是,此操作(可理解)仅显示最后一个变体:

<p><strong>(gene, position, variant, call):</strong> {{ sample.sample_variant }}

循环代码是我一直在玩的东西,但根本无法正常工作,如果我将{% render_table var %}行保留在其中,则会出现此错误:

  File "/opt/miniconda3/envs/py37/lib/python3.7/site-packages/django/template/response.py", line 83, in rendered_content
    content = template.render(context, self._request)
  File "/opt/miniconda3/envs/py37/lib/python3.7/site-packages/django/template/backends/django.py", line 61, in render
    return self.template.render(context)
  File "/opt/miniconda3/envs/py37/lib/python3.7/site-packages/django/template/base.py", line 171, in render
    return self._render(context)
  File "/opt/miniconda3/envs/py37/lib/python3.7/site-packages/django/template/base.py", line 163, in _render
    return self.nodelist.render(context)
  File "/opt/miniconda3/envs/py37/lib/python3.7/site-packages/django/template/base.py", line 937, in render
    bit = node.render_annotated(context)
  File "/opt/miniconda3/envs/py37/lib/python3.7/site-packages/django/template/base.py", line 904, in render_annotated
    return self.render(context)
  File "/opt/miniconda3/envs/py37/lib/python3.7/site-packages/django/template/loader_tags.py", line 150, in render
    return compiled_parent._render(context)
  File "/opt/miniconda3/envs/py37/lib/python3.7/site-packages/django/template/base.py", line 163, in _render
    return self.nodelist.render(context)
  File "/opt/miniconda3/envs/py37/lib/python3.7/site-packages/django/template/base.py", line 937, in render
    bit = node.render_annotated(context)
  File "/opt/miniconda3/envs/py37/lib/python3.7/site-packages/django/template/base.py", line 904, in render_annotated
    return self.render(context)
  File "/opt/miniconda3/envs/py37/lib/python3.7/site-packages/django/template/loader_tags.py", line 62, in render
    result = block.nodelist.render(context)
  File "/opt/miniconda3/envs/py37/lib/python3.7/site-packages/django/template/base.py", line 937, in render
    bit = node.render_annotated(context)
  File "/opt/miniconda3/envs/py37/lib/python3.7/site-packages/django/template/base.py", line 904, in render_annotated
    return self.render(context)
  File "/opt/miniconda3/envs/py37/lib/python3.7/site-packages/django_tables2/templatetags/django_tables2.py", line 148, in render
    raise ValueError("Expected table or queryset, not {}".format(klass))
ValueError: Expected table or queryset, not str

我知道由传递字符串而不是查询集/表类对象引起的错误,但我还没有弄清楚如何解决它。

django_tables2版本是最新版本,正在呈现其他表而没有问题。

  1. 我以与在其他模型(genesample_name)中的字段相似的方式将pi链接到列上?
  2. 如何安排for循环以正确显示列表中的所有变体?
  3. 我需要更改(假设)views.py文件以正确呈现var表吗?
  4. 我不相信我已在urls.py中正确列出了该路径(我摸索了几种变体,但均未成功),如果不正确,应该如何正确构造路径?

非常感谢和感谢。

1 个答案:

答案 0 :(得分:0)

好-非常感谢一位帮助我找出错误的朋友(AR)。

第一个问题是我的关系不正确;外键必须在Variant模型中,而不是Sample模型中(sample_variant模型中的Sample已删除)。

models.py

class Variant(models.Model):                                                              
   gene = models.CharField('Gene', max_length=100, blank=True, default='')
   nucleotide_variant = models.CharField('Variant', max_length=50, blank=True, default='')   call = models.CharField('Call', max_length=100, blank=True, default='')
   position = models.IntegerField('Position', blank=True, null=True) 
   sample_n = models.ForeignKey(Sample, on_delete=models.SET_NULL, null=True)

   class Meta:
      ordering = ('sample_n', 'gene', 'position', 'nucleotide_variant', 'call')

   def __str__(self):
      return f'{self.sample_n}, {self.gene}, {self.position}, {self.nucleotide_variant}, {self.call}'

下一件事是我将字段variant更改为nucleotide_variant。这可能是不必要的,但对我来说更清楚了。

views.py

class VariantListView(generic.ListView):
   model = Variant
   paginate_by = 100 

@login_required
def var(request):
   variant = Variant.objects.get(pk = pk) 
   table = VariantTable(Variant.objects.filter(variant=pk))
   RequestConfig(request).configure(table)                                                
   return render(request, 'samples/sample_detail.html', {'variant': variant, 'var': table}) 

urls.py

path('samples/', views.var, name='var'),

html

   <h3>Variant Results</h3>
   {% render_table sample.variant_set.all %}

问题已解决,表格已呈现。