迁移后在 Django 上更改 db_table 不起作用,并且添加用户 (AbstractUser) 外键会破坏 __str__()
Posted
技术标签:
【中文标题】迁移后在 Django 上更改 db_table 不起作用,并且添加用户 (AbstractUser) 外键会破坏 __str__()【英文标题】:Changing db_table on Django doesn't work after migrations, and adding a user (AbstractUser) foreign key breaks __str__() 【发布时间】:2021-05-05 14:06:33 【问题描述】:这是一个有 2 个问题的帖子,我怀疑它们之间存在某种联系。 我在 Django 上的帖子应用程序没有正确迁移(我认为),添加“作者”(用户 AbstractUser 模型)会破坏我的代码并出现错误:
str 返回非字符串(元组类型)
每当我尝试添加一行时。
我使用 db_table
更改了表的名称(因此它将在管理页面上将 Postss 更改为 Posts)并向 __str__
函数添加了一个输出,我尝试了 的一些输出变体str 函数,但没有一个起作用。我什至试图评论它。还是。表名或错误没有变化。我尝试更改我的 str 函数几次并得到相同的结果。删除 str 函数时,我仍然得到相同的结果。
我对这种情况有一些疯狂的猜测:
由于某种原因,它可能无法与抽象用户类一起使用,并且不知道如何解决这个问题以及为什么会发生这种情况。 (删除author=...
行可以解决此问题,但不会更改表名,但我确实需要用户 fk)。
由于我的环境或 python 版本Python 3.9.1
(?),它可能无法正确迁移。
PS:我所做的每一项更改都包括停止服务器运行和运行以下行:python manage.py migrate
和 python manage.py makemigrations
。
posts/models.py:
from django.db import models
from users.models import Users
class Posts(models.Model):
content = models.TextField(max_length=255)
author = models.ForeignKey('users.Users', on_delete=models.CASCADE, null=True)
timestamp = models.DateTimeField(auto_now_add=True, null=True)
class Meta:
db_table = 'Posts'
def __str__(self):
return self.content + " " + self.author.email + " " + self.timestamp
我也会添加我的用户模型:
from django.db import models
from django.contrib.auth.models import AbstractUser
# https://rahmanfadhil.com/django-login-with-email/
class Users(AbstractUser):
USERNAME_FIELD = 'email'
email = models.EmailField(unique=True)
REQUIRED_FIELDS = ['username'] # removes email from REQUIRED_FIELDS
def __str__(self):
return (self.email, self.username)
【问题讨论】:
您的模型用户__str__
返回tuple
,但它必须是string
。我更新了我的答案。
【参考方案1】:
您正在尝试返回 tuple
,它必须是 string
。
def str(自我): 返回(self.email,self.username)
使用Python中的新格式
试试这个:
class Users(AbstractUser):
...
def __str__(self):
template = '0.email 0.username'
return template.format(self)
class Posts(models.Model):
...
def __str__(self):
template = '0.content 0.author.email 0.timestamp'
return template.format(self)
或
def __str__(self):
return ' '.format(self.content, self.author.email, self.timestamp)
第二个问题:
如果我正确理解您的问题,那么要更改模型的标签,您需要使用verbose_name
:
class Meta:
verbose_name = 'Post'
verbose_name_plural = 'Posts'
使用选项db_table,您可以更改用于模型的数据库表的名称。
看看它是如何工作的,来自 Django:如果你有一个应用程序 bookstore
(由 manage.py startapp bookstore
创建),定义为 class Book
的模型将有一个名为 bookstore_book
的数据库表。
那么db_table
必须是:
class Meta:
# your desired table name can you find in your database
db_table = 'appLabel_modelName'
要列出 PostgreSQL 的当前数据库中的表,您可以运行:
python manage.py dbshell
\dt
另见 Django db_table
【讨论】:
是的,这行得通!我不知道这样做仍然使用用户的 str 功能。但是我的表名没有得到db_table='Posts'
的值。为什么?
@W.Doch,如果我正确理解了您的第二个问题,那么要更改模型的标签,您需要使用 verbose_name。我更新了我的答案。
所以我发现我要找的是verbose_name_plural = "Posts"
而不是db_table
... django 对我来说仍然很奇怪。谢谢你的朋友。
@W.Doch,乐于助人?
啊,我看到你编辑了你的帖子;)!【参考方案2】:
尝试将您的 __str__
方法更改为:
def __str__(self):
return self.content + " " + str(self.author.email) + " " + str(self.timestamp)
此外,您只需在更改模型而不是模型方法时进行迁移。
【讨论】:
嗨,SLEm。好的。很高兴知道。无论如何,这不起作用。我粘贴了我的错误截图,以使我的问题更清楚.. 您的User
模型伙伴实际上没有username
字段,这可能是您的User
__str__
方法中的问题?
我这样做是因为它是默认用户类的类扩展(使用 AbstractUser 类)docs.djangoproject.com/en/3.1/topics/auth/customizing/… 我只是将其更改为使用电子邮件作为主要标识符而不是用户名。以上是关于迁移后在 Django 上更改 db_table 不起作用,并且添加用户 (AbstractUser) 外键会破坏 __str__()的主要内容,如果未能解决你的问题,请参考以下文章
Django:添加整数字段后在应用程序上运行 makemigrations 时出现“未知列”
在 Heroku 上,在实例已经使用更改的模型代码重新启动后,Django syncdb / South 迁移是不是存在危险?
Django 在本地迁移工作,但不在 Elastic Beanstalk 生产上
Django 迁移错误:您无法更改 M2M 字段或从 M2M 字段更改,或通过 = 在 M2M 字段上添加或删除