3

我在 Django 中遇到了一些缓存问题。到目前为止,我仅在运行 testsuite 时才看到此问题。问题是有时(这似乎总是在第二次调用代码时发生),Django 不会更新它的缓存或者它变得不一致。

经过一些调试的提取代码是:

class Source(models.Model):
    name = models.CharField(max_length = 50)
    quality = models.IntegerField(default = 0)

class Reference(models.Model):
    url = models.URLField()
    source = models.ForeignKey(Source)

    class Meta:
        ordering = ['-source__quality']

class Issue(models.Model):
    references = models.ManyToManyField(Reference)
    master = models.ForeignKey(Reference, related_name = 'mastered_issue_set')

def auto_create(instance):
    issue = Issue.objects.create(master = instance)
    print issue.references.count(), issue.references.all()
    issue.references.add(instance)
    print issue.references.count(), issue.references.all()

在第一次调用时,我正确地得到以下输出:

0 []
1 [<Reference: test>]

然而,在第二次调用 to 时auto_create,Django 认为有一个参考,但它没有给我:

0 []
1 []

这种行为当然会破坏进一步的代码。知道这里可能出了什么问题,或者至少如何调试它?

PS:看起来Reference上课订购是造成这种情况的原因。但我仍然不清楚为什么。

4

2 回答 2

0

我无法用 sqlite3 重现。难道是Reference传入的实例没有保存?以下运行没有打嗝:

def auto_create(instance):
    issue = Issue.objects.create(master = instance)
    print issue.references.count(), issue.references.all()
    assert issue.references.count()==0, "initial ref count is not null"
    assert len(issue.references.all())==0, "initial ref array is not empty"
    issue.references.add(instance)
    print issue.references.count(), issue.references.all()
    assert issue.references.count()==1, "ref count is not incremented"
    assert len(issue.references.all())==1, "initial ref array is not populated"


def test_auto():
    s = Source()
    s.save()
    r = Reference(source=s)
    r.save()
    auto_create(r)
于 2012-12-10T23:54:40.983 回答
0

最后我找到了导致这个问题的原因。这是我自己的缓存代码,而不是 Django 的。

我有自定义源管理器,它返回并缓存了一些标准源:

class SourceManager(models.Manager):
    url_source = None
    def get_generic(self):
        if self.url_source is None:
            self.url_source, created = self.get_or_create(name = 'URL', quality = 0)
        return self.url_source

class Source(models.Model):
    name = models.CharField(max_length = 50)
    quality = models.IntegerField(default = 0)

    objects = SourceManager()

这在应用程序中工作得非常好——一旦创建了源,管理器就会记住它的存在,因为源在其生命周期内不会改变。但是在测试中它们消失了,因为整个测试在单个事务中运行然后恢复。

我觉得奇怪的是,它models.ForeignKey并没有抱怨得到不存在的对象,但后来出现了错误,而source__quality作为底层 JOIN SELECT 排序时找不到匹配的Source对象。

于 2012-12-11T08:11:46.237 回答