Python Django测试中的异常错误错误认为实例不';不存在

Python Django测试中的异常错误错误认为实例不';不存在,python,django,django-testing,django-tests,Python,Django,Django Testing,Django Tests,我有一个非常定制的Django应用程序,可以检查用户是否可以将ForeignKey更改为某些值 在这种情况下,用户属于工作组,项目也可以属于工作组,因此,当用户制作一个项目时,他们只能将其放入他们所属的工作组。复杂事项Item是一个父类,因此有很多类型的“Item” 目前,我有一个自定义管理员表单设置来检查: class AdminConceptForm(autocomplete_light.ModelForm): def __init__(self, *args, **kwargs):

我有一个非常定制的Django应用程序,可以检查用户是否可以将ForeignKey更改为某些值

在这种情况下,
用户
属于
工作组
项目
也可以属于
工作组
,因此,当
用户
制作一个
项目
时,他们只能将其放入他们所属的
工作组
。复杂事项
Item
是一个父类,因此有很多类型的“Item”

目前,我有一个自定义管理员表单设置来检查:

class AdminConceptForm(autocomplete_light.ModelForm):
    def __init__(self, *args, **kwargs):
        #... other code
        self.fields['workgroup'].queryset = self.request.user.profile.editable_workgroups.all()
此测试的重要部分包括:

def setUp(self):
    from django.test import Client

    self.client = Client()
    self.wg1 = models.Workgroup.objects.create(name="Test WG 1") # Editor is member

def test_editor_change_item(self):
    self.login_editor()
    response = self.client.get(reverse("admin:%s_%s_change"%(self.itemType._meta.app_label,self.itemType._meta.model_name),args=[self.item1.pk]))
    self.assertResponseStatusCodeEqual(response,200)

    updated_item = dict((k,v) for (k,v) in model_to_dict(self.item1).items() if v is not None)
    updated_name = updated_item['name'] + " updated!"
    updated_item['name'] = updated_name

    updated_item.update({
        'statuses-TOTAL_FORMS': 0, 'statuses-INITIAL_FORMS': 0 #no statuses
    })
    updated_item.update(self.form_defaults)
    self.assertTrue(self.wg1 in self.editor.profile.myWorkgroups)

    self.assertEqual([self.wg1],list(response.context['adminform'].form.fields['workgroup'].queryset))

    self.assertTrue(perms.user_can_edit(self.editor,self.item1))
    self.assertTrue(self.item1.workgroup in self.editor.profile.editable_workgroups.all())

    response = self.client.post(
            reverse("admin:%s_%s_change"%(self.itemType._meta.app_label,self.itemType._meta.model_name),args=[self.item1.pk]),
            updated_item
            )

# HERE IS WHERE THE FAILURE IS!!!
    self.assertResponseStatusCodeEqual(response,302)

    self.item1 = self.itemType.objects.get(pk=self.item1.pk)
    self.assertEqual(self.item1.name,updated_name)
但有时(间歇性地),当我运行测试套件时,我会
发布到该表单以测试保存内容,我会得到以下错误:

======================================================================
FAIL: test_editor_change_item (aristotle_mdr.tests.test_extension_api.QuestionAdmin)
----------------------------------------------------------------------
Traceback (most recent call last):
  File "/home/travis/build/aristotle-mdr/aristotle-metadata-registry/aristotle_mdr/tests/test_admin_pages.py", line 285, in test_editor_change_item
    self.assertResponseStatusCodeEqual(response,302)
  File "/home/travis/build/aristotle-mdr/aristotle-metadata-registry/aristotle_mdr/tests/utils.py", line 501, in assertResponseStatusCodeEqual
    self.assertEqual(response.status_code, code)
AssertionError: 200 != 302
由于这一性质,如果发布成功,页面应该重定向,我有一些代码,如果不是这样的话,只会吐出响应HTML,在这些情况下,我得到以下结果:

<label class="required" for="id_workgroup">Workgroup</label>
<select id="id_workgroup" name="workgroup">
    <option value="">---------</option>
    <option value="17" selected="selected">Test WG 1</option>
</select>
<ul class="errorlist">
    <li>workgroup instance with pk 17 does not exist.</li>
</ul>
事实上,调用admin
changelist
视图总是会导致任何后续管理员页面在尝试保存时在Postgres上失败,因为
工作组
不再出现在查询集中现在,这是为什么?

完整代码:

class MinimalExample(TestCase):
    itemType=models.ObjectClass
    form_defaults = {}
    create_defaults = {}
    def setUp(self):
        self.wg1 = models.Workgroup.objects.create(name="Test WG")

        self.editor = User.objects.create_user('eddie','','editor')
        self.editor.is_staff=True
        self.editor.save()

        self.wg1.submitters.add(self.editor)

        self.assertEqual(self.editor.profile.editable_workgroups.count(),1)
        self.item1 = self.itemType.objects.create(name="admin_page_test_oc",description=" ",workgroup=self.wg1,**self.create_defaults)
    def logout(self):
        self.client.post(reverse('django.contrib.auth.views.logout'), {})

    def login_editor(self):
        self.logout()
        response = self.client.post(reverse('friendly_login'), {'username': 'eddie', 'password': 'editor'})
        self.assertEqual(response.status_code,302)
        return response
    def assertResponseStatusCodeEqual(self,response,code):
        self.assertEqual(response.status_code, code)

    def test_bar(self):
        print("Do Bar")
        self.do_foo()
        print("Bar done")
    def test_foo(self):
        print("Do Foo")
        self.login_editor()
        response = self.client.get(reverse("admin:%s_%s_changelist"%(self.itemType._meta.app_label,self.itemType._meta.model_name)))
        self.assertResponseStatusCodeEqual(response,200)
        self.do_foo()
        print("Foo done")
    def test_zip(self):
        print("Do Zip")
        self.do_foo()
        print("Zip done")
    def do_foo(self):
        url_bits = (self.itemType._meta.app_label,self.itemType._meta.model_name)
        response = self.client.post(reverse('friendly_login'), {'username': 'eddie', 'password': 'editor'})

        response = self.client.get(reverse("admin:%s_%s_add"%url_bits))

        data = {'name':"admin_page_test_oc",'description':"test","workgroup":self.wg1.id,
                    'statuses-TOTAL_FORMS': 0, 'statuses-INITIAL_FORMS': 0 #no substatuses
                }
        response = self.client.post(reverse("admin:%s_%s_add"%url_bits),data)
        self.item1 = self.itemType.objects.first()
        response = self.client.get(reverse("admin:%s_%s_change"%url_bits,args=[self.item1.id]))

        data['name'] = "updated"
        # Re post the same data
        response = self.client.post(
                reverse("admin:%s_%s_change"%url_bits,args=[self.item1.id]),
                data
                )
        print response
        self.item1 = self.itemType.objects.first() # decache
        self.assertTrue(self.item1.name == "updated")

我先从你加载工作组的地方开始。尝试将工作组创建移出
设置
并移至
设置类
。这会使工作组在TestCase中的所有测试中保持原位,这可能是您想要的

def setUp(self):
    from django.test import Client
    self.client = Client()

@classmethod
def setUpClass(cls):
    cls.wg1 = models.Workgroup.objects.create(name="Test WG 1") # Editor is member
    super().setUpClass() # Python 3 version

固定装置通常是一场噩梦,但如果这没有帮助,我会有兴趣看看是否将该工作组移动到固定装置,看看问题是否消失。

哇,多么疯狂的旅程

原来是
RelatedListFilter
中的一个问题导致了自定义查询集出现问题。指出:

过滤器查找以某种方式被缓存

field和field.rel对象将在请求之间保持

甚至在某些情况下,跨事务回滚似乎也是如此


故事的寓意是,在管理中使用
ListFilters
时要小心

您能显示出现此错误的实际测试吗?@knbk Done和Done。如果您单独运行此特定测试方法(删除所有*.pyc文件后),是否也会发生这种情况?您的视图可能正在重用同一个表单实例,因此queryset只设置一次,永远不会更改。当然,如果您也向我们展示了您的视图代码,那么就更容易排除这种情况。@RobertJørgensgaardEngdahl这主要发生在Travis CI测试实例上,该实例每次都在新环境中执行,因此它不是陈旧的*.pyc文件。没有“视图”作为管理页面,但我可以将您链接到完整的代码。queryset的想法很好,但是你可以看到这个选项在HTML中是可见的,所以有些东西是存在的。如果我是你的话,我会去阅读Django的管理代码,看看是什么导致了这种情况。作者可能没有考虑到您的特定设置。Django是最好的,但它不是没有bug的。这可能不起作用,因为有些项目在操作过程中被删除了,但我认为工作组的情况并非如此。我可以试试看!我会让你知道的。你只有在运行完整的测试套件时才会遇到问题吗?如果是这样,你可能会有另一个测试干扰它。也许检查一下你的其他测试,如果他们删除了一些东西,把它放回一个拆卸方法中。
def setUp(self):
    from django.test import Client
    self.client = Client()

@classmethod
def setUpClass(cls):
    cls.wg1 = models.Workgroup.objects.create(name="Test WG 1") # Editor is member
    super().setUpClass() # Python 3 version