如何使 Django 模型字段从另一个字段派生其值并且不可编辑
How to make Django model field derive its value from another field and be non editable
我有一个用例,在这个用例中,我在 Django 休息框架中创建了一个自定义用户模型,另一个名为 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
在书籍模型中,我有一个名为 user 的字段,它基本上是一个外键,将创建的书链接到创建该书的人和另一个名为 author 的字段,它基本上是来自用户模型的一个名为 author_pseudonm 的字段.
我对 rest 框架还很陌生,我想知道这是否是正确的方法。
我想在创建图书时自动填充 author_pseudonm 字段,并且不可编辑。
请建议我如何使用其余框架实现相同的目标。
基本上您不想以这种方式复制您的数据 - 您不需要将 author
存储在您的 Book
模型中,因为它已经存在于您的 User
中模型。每当你需要获取给定书籍的作者笔名时,你应该简单地做
book.user.author_pseudonm
或者如果您想在 Django 管理或序列化程序或查询集过滤器中引用该列,您可以使用 user__author_pseudonm
(注意:你这里好像也有错别字author_pseudonm
-> author_pseudonym
)
如果我理解正确,你的问题是用户可以随着时间的推移更改他们的笔名,并且当作者更改他们的笔名时,特定书籍的笔名不应该改变。
我这样做的方法是使用自定义 save
函数,在给定 User
外键的情况下,获取 author_pseudonym
值并将其作为 author
字段并保存模型。您可以在 author
字段定义中放置一个 editable=False
标志,以确保将来不会有人不小心编辑它。
以下是伪代码,但我想 Book
的 save
函数看起来像这样:
def save(self, *args, **kwargs):
user = self.user
self.author = user.author_pseudonm
super(Book, self).save(**kwargs)
郑重声明,DRF 和 vanilla Django 的 models
部分是相同的。无论您是否使用 DRF,您都将以相同的方式实现这一点,因为这只与模型有关。
我有一个用例,在这个用例中,我在 Django 休息框架中创建了一个自定义用户模型,另一个名为 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
在书籍模型中,我有一个名为 user 的字段,它基本上是一个外键,将创建的书链接到创建该书的人和另一个名为 author 的字段,它基本上是来自用户模型的一个名为 author_pseudonm 的字段.
我对 rest 框架还很陌生,我想知道这是否是正确的方法。
我想在创建图书时自动填充 author_pseudonm 字段,并且不可编辑。
请建议我如何使用其余框架实现相同的目标。
基本上您不想以这种方式复制您的数据 - 您不需要将 author
存储在您的 Book
模型中,因为它已经存在于您的 User
中模型。每当你需要获取给定书籍的作者笔名时,你应该简单地做
book.user.author_pseudonm
或者如果您想在 Django 管理或序列化程序或查询集过滤器中引用该列,您可以使用 user__author_pseudonm
(注意:你这里好像也有错别字author_pseudonm
-> author_pseudonym
)
如果我理解正确,你的问题是用户可以随着时间的推移更改他们的笔名,并且当作者更改他们的笔名时,特定书籍的笔名不应该改变。
我这样做的方法是使用自定义 save
函数,在给定 User
外键的情况下,获取 author_pseudonym
值并将其作为 author
字段并保存模型。您可以在 author
字段定义中放置一个 editable=False
标志,以确保将来不会有人不小心编辑它。
以下是伪代码,但我想 Book
的 save
函数看起来像这样:
def save(self, *args, **kwargs):
user = self.user
self.author = user.author_pseudonm
super(Book, self).save(**kwargs)
郑重声明,DRF 和 vanilla Django 的 models
部分是相同的。无论您是否使用 DRF,您都将以相同的方式实现这一点,因为这只与模型有关。