我有一个用例,其中我已经在Django rest框架中创建了一个自定义用户模型,并创建了一个由用户创建的名为Book的模型。我的模型如下:
from django.db import models
from django.contrib.auth.models import (
AbstractBaseUser,
BaseUserManager,
PermissionsMixin)
from django.conf import settings
class UserManager(BaseUserManager):
def create_user(self, email, password=None, author_pseudonm=None, **extra_fields):
"""Creates and saves a new user """
if not email:
raise ValueError('Users must have an email address')
if not author_pseudonm:
raise ValueError("User must have an author pseudo name field")
user = self.model(email=self.normalize_email(email),
author_pseudonm=author_pseudonm, **extra_fields)
user.set_password(password)
user.save(using=self._db)
return user
def create_superuser(self, email, password, author_pseudonm):
"""Creates and saves a new super user"""
user = self.create_user(
email=email, password=password, author_pseudonm=author_pseudonm)
user.is_staff = True
user.is_superuser = True
user.save(using=self._db)
return user
class User(AbstractBaseUser, PermissionsMixin):
"""Custom user model which supports using email instead of username"""
email = models.EmailField(max_length=255, unique=True)
name = models.CharField(max_length=255)
author_pseudonm = models.CharField(max_length=255)
is_active = models.BooleanField(default=True)
is_staff = models.BooleanField(default=False)
objects = UserManager()
USERNAME_FIELD = 'email'
REQUIRED_FIELDS = ["author_pseudonm"]
class Book(models.Model):
"""Custom book model"""
title = models.CharField(max_length=255, unique=True)
user = models.ForeignKey(
settings.AUTH_USER_MODEL,
on_delete=models.CASCADE
)
description = models.TextField()
price = models.IntegerField()
author = models.CharField(default=user.author_pseudonm)
is_published = models.BooleanField(default=True)
def __str__(self):
return self.title
在书模型中,我有一个名为“用户”的字段,该字段基本上是将创建的书链接到创建该书的人的外键,还有一个名为“作者”的字段,该字段基本上是用户模型中的一个名为author_pseudonm的字段。
我刚开始接触框架,我想知道这是否是正确的方法。
我想在创建书籍时自动填充author_pseudonm字段,并且该字段不可编辑。
请提出我如何使用其余框架实现相同的目标。
答案 0 :(得分:2)
基本上,您不想以这种方式复制数据-您不需要将author
存储在Book
模型中,因为它已经存在于User
模型中。每当您需要获得给定书的作者笔名时,您只需
book.user.author_pseudonm
,或者如果您想引用Django管理员或序列化程序或queryset过滤器中的列,则可以使用user__author_pseudonm
(注意:您在这里似乎也有错字author_pseudonm
-> author_pseudonym
)
答案 1 :(得分:2)
如果我对您的理解正确,那么您的问题是用户可以随时间更改其笔名,而当作者更改其笔名时,特定书籍的笔名也不应更改。
我这样做的方式是拥有一个自定义save
函数,给定User
外键,获取author_pseudonym
值并将其输入为{{1 }}字段并保存模型。您可以在author
字段定义中放置editable=False
标志,以确保将来不会有人意外编辑它。
以下是伪代码,但我想您的author
的{{1}}函数看起来像这样:
save
为便于记录,DRF的Book
部分和香草Django是相同的。无论是否使用DRF,您都可以以相同的方式实现此目的,因为这仅与模型有关。