django:如何为工厂男孩动态指定数据库

django: how to specify database dynamically for factory boy

我正在设置一个包含大量数据库的 Django 应用程序,其中一些正在使用相同的模型(它们不是副本)。我已经配置了我的路由器并且一切正常。当我想使用 factory-boy 进行测试时出现问题。

在另一个项目中,我可以在 Meta 中设置数据库,但现在,我必须 select 在哪个数据库上动态创建实例(如果没有,我将不得不创建一个 DjangoModelFactory 每个数据库,这不会很漂亮)。

有没有一种(更简单的)方法可以为每次创建动态指定数据库?

据我所知 factory_boy (version <=2.10.0) 没有提供类似的东西。

不过,您的问题是使用上下文管理器的完美用例。它将允许您在任何需要的地方动态设置数据库,并且只在所需的范围内,并且 DRY!:

# factoryboy_utils.py

@classmethod
def _get_manager(cls, model_class):
    return super(cls, cls)._get_manager(model_class).using(cls.database)

class DBAwareFactory(object):
    """
    Context manager to make model factories db aware

    Usage:
        with DBAwareFactory(PersonFactory, 'db_qa') as personfactory_on_qa:
            person_on_qa = personfactory_on_qa()
            ...
    """
    def __init__(self, cls, db):
        # Take a copy of the original cls
        self.original_cls = cls
        # Patch with needed bits for dynamic db support
        setattr(cls, 'database', db)
        setattr(cls, '_get_manager', _get_manager)
        # save the patched class
        self.patched_cls = cls

    def __enter__(self):
        return self.patched_cls

    def __exit__(self, type, value, traceback):
        return self.original_cls

然后,在您的测试中,您可以执行以下操作:

from factoryboy_utils import DBAwareFactory

class MyTest(TestCase):

   def test_mymodel_on_db1(self):
       ...
       with DBAwareFactory(MyModelFactory, 'db1') as MyModelFactoryForDB1:
           mymodelinstance_on_db1 = MyModelFactoryForDB1()
           # whatever you need with that model instance
           ...
       # something else here

   def test_mymodel_on_db2(self):
       ...
       with DBAwareFactory(MyModelFactory, 'db2') as MyModelFactoryForDB2:
           mymodelinstance_on_db2 = MyModelFactoryForDB2()
           # whatever you need with that model instance
           ...
       # something else here