在Django 1.5中使用自定义用户模型作为ForeignKey进行测试

时间:2013-03-12 19:07:11

标签: python django django-testing django-1.5

我正在使用Django 1.5,我正在努力使应用程序适用于任何自定义用户模型。我已将应用程序更改为在任何地方使用get_user_model,并且应用程序本身到目前为止没有显示任何问题。

问题是我希望能够测试应用程序,但我找不到使用自定义用户模型正确测试ForeignKey模型字段的方法。当我运行下面附带的测试用例时,我收到此错误:

ValueError: Cannot assign "<NewCustomUser: alice@bob.net>": "ModelWithForeign.user" must be a "User" instance.

这是我用于测试的文件:

from django.conf import settings
from django.contrib.auth import get_user_model
from django.contrib.auth.tests.custom_user import CustomUser, CustomUserManager
from django.db import models
from django.test import TestCase
from django.test.utils import override_settings

class NewCustomUser(CustomUser):
    objects = CustomUserManager()
    class Meta:
        app_label = 'myapp'

class ModelWithForeign(models.Model):
    user = models.ForeignKey(settings.AUTH_USER_MODEL)

@override_settings(
    AUTH_USER_MODEL = 'myapp.NewCustomUser'
)   
class MyTest(TestCase):
    user_info = {
        'email': 'alice@bob.net',
        'date_of_birth': '2013-03-12',
        'password': 'password1'
    }   

    def test_failing(self):
        u = get_user_model()(**self.user_info)
        m = ModelWithForeign(user=u)
        m.save()

我正在引用ForeignKey参数列表中的用户模型,如here所述,但使用get_user_model并没有改变任何内容,因为user属性是在设置更改发生之前进行评估。当我使用自定义用户模型时,有没有办法让这个ForeignKey玩得很好?

1 个答案:

答案 0 :(得分:3)

我也在Django邮件列表上询问了这一点,但似乎至少在目前,没有办法更改settings.AUTH_USER_MODEL并让它与ForeignKey很好地协作。

到目前为止,为了测试我的应用,我已经从this answer创建了一个runtests.py文件:

import os, sys
from django.conf import settings

if len(sys.argv) >= 2:
    user_model = sys.argv[1]
else:
    user_model = 'auth.User'

settings.configure(
    ...
    AUTH_USER_MODEL=user_model,
    ...
)

...

并添加了一个bash脚本,以便使用不同的用户模型实际运行测试:

for i in "auth.User" "myapp.NewCustomUser"; do
    echo "Running with AUTH_USER_MODEL=$i"
    python runtests.py $i
    if [ $? -ne 0 ]; then
        break
    fi
done

最后一点是使用函数来实际检索正确的用户模型信息,而不是仅仅使用“静态”变量:

def get_user_info():
    if settings.AUTH_USER_MODEL == 'auth.User':
        return {default user info}
    if settings.AUTH_USER_MODEL == 'myapp.NewCustomUser':
        return {my custom user info}
    raise NotImplementedError

我并不认为这是对这个问题的正确答案,但到目前为止......它确实有效。