gpt4 book ai didi

postgresql - 如何防止 sql alchemy 将 None 值插入字段?

转载 作者:行者123 更新时间:2023-11-29 12:19:05 29 4
gpt4 key购买 nike

Alembic 迁移脚本:

def upgrade():
uuid_gen = saexp.text("UUID GENERATE V1MC()")
op.create_table(
'foo',
sa.Column('uuid', UUID, primary_key=True, server_default=uuid_gen),
sa.Column(
'inserted',
sa.DateTime(timezone=True),
server_default=sa.text("not null now()"))
sa.Column('data', sa.Text)
)

这是我的 SQL Alchemy 基类:

Class Foo(Base):
__tablename__ = 'foo'
inserted = Column(TIMESTAMP)
uuid = Column(UUID, primary_key=True)
data = Column(TEXT)

它有一个用于插入的静态方法:

@staticmethod
def insert(session, jsondata):
foo = Foo()
foo.data = jsondata['data']
if 'inserted' in jsondata:
foo.inserted = jsondata['inserted']
if 'uuid' in jsondata:
foo.uuid = jsondata['uuid']
session.add(foo)
return foo

2 个 if 的目的是简化测试。这样我就可以“注入(inject)”一个uuid和插入的日期,为我的测试获取可预测的数据

尝试插入数据时

foo = Foo()
foo.insert(session, {"data": "foo bar baz"})
session.commit()

我得到一个 IntegrityError :

[SQL: 'INSERT INTO foo (inserted, data) VALUES (%(inserted)s, %(data)s) RETURNING foo.uuid'] [parameters: {'data': 'foo bar baz', 'inserted': None}]

这对我来说很正常,因为插入违反了 postgres 数据库中的“非空”约束。

如何防止 sql alchemy 将 None 值插入到插入的字段中?

在玩和测试时,我发现如果“插入”列定义为主键,sql alchemy 不会在插入语句中包含该字段。

def upgrade():
uuid_gen = saexp.text("UUID GENERATE V1MC()")
op.create_table(
'foo',
sa.Column('uuid', UUID, primary_key=True, server_default=uuid_gen),
sa.Column(
'inserted',
primary_key=True,
sa.DateTime(timezone=True),
server_default=sa.text("not null now()"))
sa.Column('data', sa.Text)
)

但这不是我想要的。

最佳答案

主要问题是 server_default,它在 Foo 类的 inserted 成员中丢失。它只存在于 alembic 脚本中。请注意,运行迁移时使用 alembic 定义。它们不会影响应用程序。因此,最好将完全相同的定义从 alembic 脚本复制到您的应用程序(反之亦然)。

因为在模型定义中没有定义任何值,sqlalchemy 似乎在实例化类时将其设置为 None。然后这将被发送到将提示的数据库。要解决此问题,请在模型定义(继承自 Base 的类)上设置 defaultserver_default

一些额外的注意事项/问题:

  • UUID GENERATE V1MC() 从哪里来? official docs看起来不一样。我将其替换为 func.uuid_generate_v1mc()
  • 您案例中的 server_default 值包含 not null,这是不正确的。您应该在列属性上设置 nullable=False(见下文)。

alembic 脚本

# revision identifiers, used by Alembic.
revision = THIS_IS_DIFFERENT_ON_EACH_INSTANCE! # '1b7e145f2138'
down_revision = None
branch_labels = None
depends_on = None

from alembic import op
import sqlalchemy as sa
from sqlalchemy.dialects.postgresql import UUID


def upgrade():
op.create_table(
'foo',
sa.Column('uuid', UUID, primary_key=True,
server_default=sa.func.uuid_generate_v1mc()),
sa.Column(
'inserted',
sa.DateTime(timezone=True),
nullable=False,
server_default=sa.text("now()")),
sa.Column('data', sa.Text)
)


def downgrade():
op.drop_table('foo')

测试器.py

from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy import Column, create_engine, func
from sqlalchemy.orm import scoped_session, sessionmaker
from sqlalchemy.dialects.postgresql import (
TEXT,
TIMESTAMP,
UUID,
)

engine = create_engine('postgresql://michel@/michel')
Session = scoped_session(sessionmaker(autocommit=False,
autoflush=False,
bind=engine))
Base = declarative_base()


class Foo(Base):
__tablename__ = 'foo'
inserted = Column(TIMESTAMP, nullable=False,
server_default=func.now())
uuid = Column(UUID, primary_key=True,
server_default=func.uuid_generate_v1mc()),
data = Column(TEXT)

@staticmethod
def insert(session, jsondata):
foo = Foo()
foo.data = jsondata['data']
if 'inserted' in jsondata:
foo.inserted = jsondata['inserted']
if 'uuid' in jsondata:
foo.uuid = jsondata['uuid']
session.add(foo)
return foo


if __name__ == '__main__':
session = Session()
Foo.insert(session, {"data": "foo bar baz"})
session.commit()
session.close()

执行后输出

[9:43:54] michel@BBS-nexus  [1 background job(s)] 
/home/users/michel/tmp› psql -c "select * from foo"
uuid | inserted | data
--------------------------------------+-------------------------------+-------------
71f5fd32-0602-11e6-aebb-27be4bbac26e | 2016-04-19 09:43:45.297191+02 | foo bar baz
(1 row)

关于postgresql - 如何防止 sql alchemy 将 None 值插入字段?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36709979/

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