迁移后在 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 migratepython 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 字段上添加或删除

更改模型以添加“通过”关系以订购多对多字段 - Django 1.7 迁移修改

Django迁移在Heroku上无法正常工作[重复]