gpt4 book ai didi

python - 相互依赖的一对多关系 SQLAlchemy

转载 作者:搜寻专家 更新时间:2023-10-30 19:56:57 25 4
gpt4 key购买 nike

我正在尝试让以下模型协同工作。首先场景如下:

  1. 一个用户可以有多个邮箱,但每个邮箱只能关联一个用户;
  2. 每个用户只能有一个主要电子邮件地址(将其视为他们当前的电子邮件地址)。

电子邮件地址是用户的 ID,因此他们必须始终拥有一个,但当他们更改它时,我想跟踪他们过去使用过的其他地址。到目前为止,设置是有一个帮助表user_emails,它在电子邮件和用户之间保持联系,我听说在使用声明性 SQLAlchemy 方法时不应该将其设置为一个类(尽管我不知道为什么)。另外,我是否认为我需要使用 use_alter=True 因为 users 表不知道外键 email_id 直到它插入?

models.py 看起来像这样:

"""models.py"""
user_emails = Table('user_emails', Base.metadata,
Column('user_id', Integer, ForeignKey('users.id'),
primary_key=True),
Column('email', String(50), ForeignKey('emails.address'),
primary_key=True))

class User(Base):
__tablename__ = 'users'
id = Column(Integer, Sequence('usr_id_seq', start=100, increment=1),
primary_key=True)
email_id = Column(String(50),
ForeignKey('emails.address', use_alter=True, name='fk_email_id'),
unique=True, nullable=False)
first = Column(String(25), unique=True, nullable=False)
last = Column(String(25), unique=True, nullable=False)

def __init__(self, first, last):
self.first = first
self.last = last

class Email(Base):
__tablename__ = 'emails'
address = Column(String(50), unique=True, primary_key=True)
user = relationship(User, secondary=user_emails, backref='emails')
added = Column(DateTime, nullable=False)
verified = Column(Boolean, nullable=False)

def __init__(self, address, added, verified=False):
self.address = address
self.added = added
self.verified = verified

在我尝试提交给数据库之前,一切似乎都很好:

>>> user = models.User("first", "last")
>>> addy = models.Email("example@example.com", datetime.datetime.utcnow())
>>> addy
<Email 'example@example.com' (verified: False)>
>>> user
>>> <User None (active: True)>
>>>
>>> user.email_id = addy
>>> user
>>> <User <Email 'example@example.com' (verified: False)> (active: True)>
>>> Session.add_all([user, addy])
>>> Session.commit()
>>> ...
>>> sqlalchemy.exc.ProgrammingError: (ProgrammingError) can't adapt type 'Email' "INSERT INTO users (id, email_id, first, last, active) VALUES (nextval('usr_id_seq'), %(email_id)s, %(first)s, %(last)s, %(active)s) RETURNING users.id" {'last': 'last', 'email_id': <Email 'example@example.com' (verified: False)>, 'active': True, 'first': 'first'}

所以,我认为我做错了什么/愚蠢的事情,但我是 SQLAlchemy 的新手,所以我不确定我需要做什么才能正确设置模型。

最后,假设我得到了正确的模型设置,是否可以添加一个关系,以便通过加载任意电子邮件对象,我将能够从电子邮件对象中的属性访问拥有它的用户?

谢谢!

最佳答案

您已经有了一个非常好的解决方案,一个小的修复将使您的代码正常工作。在下面找到对您的代码的快速反馈:

  • 您需要 use_alter=True 吗?不,您实际上不需要那个。如果 Email 表的 primary_key 是在数据库级别计算的(与基于 autoincrement 的主键一样),那么您可能需要它你有两个表,彼此之间有 foreign_keys。在你的情况下,你甚至没有那个,因为你有第三个表,所以对于任何关系组合,SA (sqlalchemy) 将通过插入新电子邮件、然后是用户、然后是关系来解决。
  • 您的代码有什么问题?:嗯,您正在将 Email 的实例分配给 User.email_id,它应该得到仅限 email 值。有两种方法可以修复它:
    1. 直接分配电子邮件。所以将 user.email_id = addy 行更改为 user.email_id = addy.address
    2. 创建关系,然后进行分配(参见下面的代码)。就个人而言,我更喜欢选项 2。
  • 其他:您当前的模型不会检查 User.email_id 实际上是 User.emails 之一。这可能是设计使然,但也可以将 [users.id, users.email_id] 中的 ForeignKey 添加到 [user_emails.user_id, user_emails.email]

版本 2 的示例代码:

""" models.py """
class User(Base):
__tablename__ = 'users'
# ...
email_id = Column(String(50),
ForeignKey('emails.address', use_alter=True,
name='fk_email_id'), unique=True,
nullable=False)
default_email = relationship("Email", backref="default_for_user")

""" script """
# ... (all that you have below until next line)
# user.email_id = addy.address
user.default_email = addy

关于python - 相互依赖的一对多关系 SQLAlchemy,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10289242/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com