2

我有一个包含两种方法的测试类,并希望在两种方法之间共享一个保存的模型实例。

我的装置:

@pytest.fixture(scope='class')
def model_factory():
    class ModelFactory(object):
        def get(self):
            x = Model(email='test@example.org',
                      name='test')
            x.save()
            return x
    return ModelFactory()

@pytest.fixture(scope='class')
def model(model_factory):
    m = model_factory.get()
    return m

我的期望是只接收model(两种)我的测试方法上的固定装置,并让它保持不变,并保留在数据库中:

@pytest.mark.django_db
class TestModel(object):

    def test1(self, model):
        assert model.pk is not None
        Model.objects.get(pk=model.pk)  # Works, instance is in the db

    def test2(self, model):
        assert model.pk is not None     # model.pk is the same as in test1
        Model.objects.get(pk=model.pk)  # Fails:
        # *** DoesNotExist: Model matching query does not exist

我已经--pdb在 结束时使用它进行了验证test1,运行Model.objects.all()返回我创建的单个实例。同时, psql 显示没有记录:

test_db=# select * from model_table;
 id | ··· fields
(0 rows)

Model.objects.all()在末尾运行in pdb 会test2返回一个空列表,考虑到该表为空,这可能是正确的。

  1. 为什么我的模型没有被持久化,而查询仍然返回一个实例?
  2. model如果我的夹具被标记scope='class'并保存,为什么在第二个测试中查询没有返回实例?(这是我最初的问题,直到我发现保存模型对数据库没有任何作用)

使用django 1.6.1, pytest-django 2.9.1,pytest 2.8.5

谢谢

4

3 回答 3

1

测试必须相互独立。为了确保这一点,Django - 像大多数框架一样 - 在每次测试后清除数据库。请参阅文档

于 2016-03-08T14:22:01.950 回答
0

Scope 参数在这种情况下有点误导,但是如果您要这样编写代码:

@pytest.fixture(scope='class')
    def model_factory(db, request):
        # body

那么你会得到一个错误,基本上说数据库夹具必须用“功能”范围实现。

我想补充一点,这目前正在开发中,将来可能会成为一个致命的功能;) github pull request

于 2016-04-19T20:46:45.147 回答
0

通过查看 postgres 日志,我发现pytest-django默认情况下ROLLBACK,每次测试后都会执行一次以保持干净(这是有道理的,因为测试不应该依赖于可能被早期测试修改的状态)。

通过装饰测试类,django_db(transaction=True)我确实可以看到 psql 在每次测试结束时提交的数据,这回答了我的第一个问题。

和以前一样,测试运行器确保在测试之间不保留任何状态,这是我第二点的答案。

于 2016-03-08T14:25:28.757 回答