迁移后在 Django 上更改 db_table 不起作用,添加用户 (AbstractUser) 外键中断 __str__()
Changing db_table on Django doesn't work after migrations, and adding a user (AbstractUser) foreign key breaks __str__()
这是一个 2 问题 post,我怀疑它们之间存在某种联系。
我在 Django 上的 posts 应用程序无法正确迁移(我认为),添加 'author'(Users AbstractUser 模型)会破坏我的代码并出现错误:
str returned non-string (type tuple)
每当我尝试添加一行时。
我使用 db_table
更改了 table 的名称(因此它将在管理页面上将 Postss 更改为 Posts)并向 __str__
函数添加了一个输出,我试过了str 函数的一些输出变体,但没有一个起作用。我什至试图将其注释掉。还是。 table 名称或错误没有变化。我尝试多次更改 str 函数并得到相同的结果。删除 str 函数时,我仍然得到相同的结果。
我对这种情况有一些疯狂的猜测:
- 出于某种原因,它可能不适用于抽象用户 class,并且不知道如何解决这个问题以及为什么会发生这种情况。 (删除
author=...
行可以解决此问题,但不会更改 table 名称,但我确实需要那里的用户 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__
方法更改为:
def __str__(self):
return self.content + " " + str(self.author.email) + " " + str(self.timestamp)
此外,您只需要在更改 模型 而不是 模型方法 .
时进行迁移
您正在尝试return一个tuple
,它必须是一个string
。
def str(self):
return (self.email, self.username)
中使用新式格式
试试这个:
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 您可以更改 数据库的名称 table 以用于模型 .
并查看它是如何工作的解释,来自 Django:如果你有一个应用程序 bookstore
(由 manage.py startapp bookstore
创建),定义为 class Book
的模型将有一个数据库 table 名为 bookstore_book
.
那么db_table
一定是:
class Meta:
# your desired table name can you find in your database
db_table = 'appLabel_modelName'
要为 PostgreSQL 列出当前数据库中的 table,您可以 运行:
python manage.py dbshell
\dt
另见 Django db_table
这是一个 2 问题 post,我怀疑它们之间存在某种联系。 我在 Django 上的 posts 应用程序无法正确迁移(我认为),添加 'author'(Users AbstractUser 模型)会破坏我的代码并出现错误:
str returned non-string (type tuple)
每当我尝试添加一行时。
我使用 db_table
更改了 table 的名称(因此它将在管理页面上将 Postss 更改为 Posts)并向 __str__
函数添加了一个输出,我试过了str 函数的一些输出变体,但没有一个起作用。我什至试图将其注释掉。还是。 table 名称或错误没有变化。我尝试多次更改 str 函数并得到相同的结果。删除 str 函数时,我仍然得到相同的结果。
我对这种情况有一些疯狂的猜测:
- 出于某种原因,它可能不适用于抽象用户 class,并且不知道如何解决这个问题以及为什么会发生这种情况。 (删除
author=...
行可以解决此问题,但不会更改 table 名称,但我确实需要那里的用户 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__
方法更改为:
def __str__(self):
return self.content + " " + str(self.author.email) + " " + str(self.timestamp)
此外,您只需要在更改 模型 而不是 模型方法 .
时进行迁移您正在尝试return一个tuple
,它必须是一个string
。
中使用新式格式def str(self): return (self.email, self.username)
试试这个:
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 您可以更改 数据库的名称 table 以用于模型 .
并查看它是如何工作的解释,来自 Django:如果你有一个应用程序 bookstore
(由 manage.py startapp bookstore
创建),定义为 class Book
的模型将有一个数据库 table 名为 bookstore_book
.
那么db_table
一定是:
class Meta:
# your desired table name can you find in your database
db_table = 'appLabel_modelName'
要为 PostgreSQL 列出当前数据库中的 table,您可以 运行:
python manage.py dbshell
\dt
另见 Django db_table