模范妈妈:多个食谱与单个食谱有外键关系

Bja*_*son 5 testing django model-mommy

我对 ModelMommy 的这种烦恼已经有一段时间了,但我不知道如何正确地做到这一点。

让我们假设一个简单的关系:

class Organization(models.Model):
    label = models.CharField(unique=True)

class Asset(models.Model):
    organization = models.ForeignKey(Organization)
    label = models.CharField(unique=True)
Run Code Online (Sandbox Code Playgroud)

和食谱:

from model_mommy.recipe import Recipe, foreign_key


organization_recipe = Recipe(Organization, label='My Organization') 
asset1_recipe = Recipe(Asset,
                      organization=foreign_key(organization_recipe),
                      label='asset 1')
asset2_recipe = Recipe(Asset,
                      organization=foreign_key(organization_recipe),
                      label='asset 2')
Run Code Online (Sandbox Code Playgroud)

现在,当我制作这些资产配方时,我收到错误:

>> asset1 = asset1_recipe.make()
>> asset2 = asset2_recipe.make()
IntegrityError: duplicate key value violates unique constraint "organizations_organization_label_key"
DETAIL:  Key (label)=(My Organization) already exists.
Run Code Online (Sandbox Code Playgroud)

这可以通过将 asset1 的组织作为参数提供给 asset2 的 make 方法来解决:

>> asset1 = asset1_recipe.make()
>> asset2 = asset2_recipe.make(organization=asset1.organization)
Run Code Online (Sandbox Code Playgroud)

但必须有一种更简单、更干净的方法来做到这一点。

编辑

根据 Helgi 的答案中的链接,我更改了所有配方外键以指向闭包:

def organization_get_or_create(**kwargs):
    """
    Returns a closure with details of the organization to be fetched from db or
    created. Must be a closure to ensure it's executed within a test case
    Parameters
    ----------
    kwargs
        the details of the desired organization
    Returns
    -------
    Closure
        which returns the desired organization
    """

    def get_org():
        org, new = models.Organization.objects.get_or_create(**kwargs)
        return org

    return get_org

my_org = organization_get_or_create(label='My Organization')

asset1_recipe = Recipe(Asset,
                      organization=my_org,
                      label='asset 1')
asset2_recipe = Recipe(Asset,
                      organization=my_org,
                      label='asset 2')
Run Code Online (Sandbox Code Playgroud)

并且可以创建任意数量的资产:

>> asset1 = asset1_recipe.make()
>> asset2 = asset2_recipe.make()
Run Code Online (Sandbox Code Playgroud)

Hel*_*lgi 0

这似乎不可能(至少现在是这样),但这个功能已经被讨论过。请参阅此处:引用相同的foreign_key对象