通過 Django 工具查找 User 數(shù)據(jù)模型
獲取 User 類的最佳方式:
# Stock user model definition
>>> from django.contrib.auth import get_user_model
>>> get_user_model()
<class 'django.contrib.auth.models.User'>
# When the project has a custom user model definition
>>> from django.contrib.auth import get_user_model
>>> get_user_model()
<class 'profiles.models.UserProfile'>
在數(shù)據(jù)模型中,使用 settings.AUTH_USER_MODEL 進(jìn)行外鍵引用
到 User 的 ForeignKey、OneToOneKey 或 ManyToManyField 的引用方式:
from django.conf import settings
from django.db import models
class IceCreamStore(models.Model):
owner = models.OneToOneField(settings.AUTH_USER_MODEL)
title = models.CharField(max_length=255)
setting.AUTH_USER_MODEL 的值一旦設(shè)置后,不要再修改,對它的修改涉及到數(shù)據(jù)庫模式的大量修改!
到 User 的外鍵引用不能用 get_user_model(),會引起死循環(huán)
以下是錯誤的使用方法:
# DON'T DO THIS!
from django.db import models
from django.contrib.auth import get_user_model
class IceCreamStore(models.Model):
# This following line tends to create import loops.
owner = models.OneToOneField(get_user_model())
title = models.CharField(max_length=255)
從 1.5- 版本的 User 數(shù)據(jù)模型遷移到 1.5+ 版本
參考: Tobias McNulty’s Tutorial 和 django-authtool’s Tutorial
Django 1.8 中自定義 User 項
相關(guān)包: django.authtools,其中有 AbstractEmailUser 和 AbstractNamedUser 數(shù)據(jù)模型。
方式 1:繼承 AbstractUser
假如只需增加額外的數(shù)據(jù)項,選中該方式。這種情況下,使用 django-authtools 中的其本數(shù)據(jù)模型、表單和 admin 對象是最快最容易的實現(xiàn)方式,而大多數(shù)項目都符合這種情況。
示例:
# profiles/models.py
from django.contrib.auth.models import AbstractUser
from django.db import models
from django.utils.translation import ugettext_lazy as _
class KarmaUser(AbstractUser):
karma = models.PositiveIntegerField(verbose_name=_("karma"),
default=0,
blank=True)
之后,在配置文件中再進(jìn)行設(shè)置:
AUTH_USER_MODEL = "profiles.KarmaUser"
方式 2:繼承 AbstractBaseUser
AbstractBaseUser 是基本骨架,只有 3 個項: password, last_login, is_active。如果符合以下情況,選擇該方式:
- 無需默認(rèn)提供的 first_name 和 last_name 等項
- 只想使用默認(rèn)的密碼處理功能
從相關(guān)的數(shù)據(jù)模型中進(jìn)行鏈接
它和 1.5 版本前創(chuàng)建 Profile 數(shù)據(jù)模型的方式類似。它有以下幾種使用情境:
情境 1: 創(chuàng)建一個第三方包
- 該包要發(fā)布到 PyPI
- 該包需為每個用戶存儲額外的信息,如 Strip ID 等
情境 2: 內(nèi)部項目需求
- 不同的用戶需要不同的項
- 該方式可以與上面的 方式 1 和方式 2 一起使用
使用這種方式,需要將不同的數(shù)據(jù)模型定義在不同的 Profile 中,如:
# profiles/models.py
from django.conf import settings
from django.db import models
from flavors.models import Flavor
class EaterProfile(models.Model):
# Default user profile
# If you do this you need to either have a post_save signal or
# redirect to a profile_edit view on initial login.
user = models.OneToOneField(settings.AUTH_USER_MODEL)
favorite_ice_cream = models.ForeignKey(Flavor, null=True, blank=True)
class ScooperProfile(models.Model):
user = models.OneToOneField(settings.AUTH_USER_MODEL)
scoops_scooped = models.IntegerField(default=0)
class InventorProfile(models.Model)
user = models.OneToOneField(settings.AUTH_USER_MODEL) .
flavors_invented = models.ManyToManyField(Flavor, null=True, blank=True)
之后,可以通過 ORM 進(jìn)行訪問, 如: user.eaterprofile.favorite_ice_cream。
注: user.get_profile() 已在 Django 1.7 中去除。
參考文獻(xiàn): Two Scoops of Django: Best Practices for Django 1.8