如何使Django模型字段从另一个字段派生其值并且不可编辑

时间:2019-10-24 19:33:37

标签: python django django-rest-framework django-orm

我有一个用例,其中我已经在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字段,并且该字段不可编辑。

请提出我如何使用其余框架实现相同的目标。

2 个答案:

答案 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,您都可以以相同的方式实现此目的,因为这仅与模型有关。