从SQlite切换后,无法在Postgresql for Flask Sqlalchemy中插入NULL作为主键值

时间:2017-07-24 21:18:49

标签: postgresql sqlite sqlalchemy flask-sqlalchemy

我最近开始将一个SQLite数据库移植到PostGreSQL上,用于使用SQLAlchemy构建的Flask网站。我在PGSQL中有我的模式,甚至将数据插入数据库。但是,我无法运行我通常的INSERT命令来向数据库添加信息。通常,我使用SQL Alchemy插入新记录,方法是将ID列保留为NULL,然后只设置其他列。但是,这会导致以下错误:

sqlalchemy.exc.IntegrityError: (psycopg2.IntegrityError) null value in column "id" violates not-null constraint
DETAIL:  Failing row contains (null, 2017-07-24 20:40:37.787393+00, 2017-07-24 20:40:37.787393+00, episode_length_list = [52, 51, 49, 50, 83]

sum_length = 0

for ..., 0, f, 101, 1, 0, 0, , null).
 [SQL: 'INSERT INTO submission (date_created, date_modified, code, status, correct, assignment_id, course_id, user_id, assignment_version, version, url) VALUES (CURRENT_TIMESTAMP, CURRENT_TIMESTAMP, %(code)s, %(status)s, %(correct)s, %(assignment_id)s, %(course_id)s, %(user_id)s, %(assignment_version)s, %(version)s, %(url)s) RETURNING submission.id'] [parameters: {'code': 'episode_length_list = [52, 51, 49, 50, 83]\n\nsum_length = 0\n\nfor episode_length in episode_length_list:\n    pass\n\nsum_length = sum_length + episode_length\n\nprint(sum_length)\n', 'status': 0, 'correct': False, 'assignment_id': 101, 'course_id': None, 'user_id': 1, 'assignment_version': 0, 'version': 0, 'url': ''}]

这是我的SQL Alchemy表声明:

class Base(Model):
    __abstract__  = True
    @declared_attr
    def __tablename__(cls):
        return cls.__name__.lower()
    def __repr__(self):
        return str(self)

    id =  Column(Integer(), primary_key=True)
    date_created  = Column(DateTime, default=func.current_timestamp())
    date_modified = Column(DateTime, default=func.current_timestamp(),
                                     onupdate=func.current_timestamp())

class Submission(Base):
    code = Column(Text(), default="")
    status = Column(Integer(), default=0)
    correct = Column(Boolean(), default=False)
    assignment_id = Column(Integer(), ForeignKey('assignment.id'))
    course_id = Column(Integer(), ForeignKey('course.id'))
    user_id = Column(Integer(), ForeignKey('user.id'))
    assignment_version = Column(Integer(), default=0)
    version = Column(Integer(), default=0)
    url = Column(Text(), default="")

我通过在脚本中调用db.create_all()来创建架构。

检查PostGreSQL方面,我们可以看到构造的表:

                                     Table "public.submission"
       Column       |           Type           | Modifiers | Storage  | Stats target | Description
--------------------+--------------------------+-----------+----------+--------------+-------------
 id                 | bigint                   | not null  | plain    |              |
 date_created       | timestamp with time zone |           | plain    |              |
 date_modified      | timestamp with time zone |           | plain    |              |
 code               | text                     |           | extended |              |
 status             | bigint                   |           | plain    |              |
 correct            | boolean                  |           | plain    |              |
 assignment_id      | bigint                   |           | plain    |              |
 user_id            | bigint                   |           | plain    |              |
 assignment_version | bigint                   |           | plain    |              |
 version            | bigint                   |           | plain    |              |
 url                | text                     |           | extended |              |
 course_id          | bigint                   |           | plain    |              |
Indexes:
    "idx_16881_submission_pkey" PRIMARY KEY, btree (id)
Foreign-key constraints:
    "submission_course_id_fkey" FOREIGN KEY (course_id) REFERENCES course(id)
    "submission_user_id_fkey" FOREIGN KEY (user_id) REFERENCES "user"(id)
Has OIDs: no

我还是新手,但不应该有序列吗?

任何关于下一步寻找什么的见解或建议都将受到超级赞赏。

2 个答案:

答案 0 :(得分:3)

标准SQL是PRIMARY KEY UNIQUENOT NULL。 PostgreSQL强制执行标准,并且允许您在表上拥有任何(不是一个)NULL。其他数据库允许您拥有一个 NULL,因此,行为也不同。

PostgreSQL current documentation on Primary Keys明确指出:

  

<强> 5.3.4。主键

     

主键约束表示列或列组可用作表中行的唯一标识符。这要求值既是唯一的又不是空的

如果您希望PRIMARY KEY是合成(即非自然)序列号,则应使用BIGSERIAL类型而不是BIGINT来定义它。我不知道使用SQLAlchemy如何实现这一点的细节,但请查看参考资料。

当您INSERT进入表格时,id NOT 应该在INSERT列列表中(不应该设置为空,只是在那里)。即:

这将生成一个新的id:

INSERT INTO public.submission (code) VALUES ('Some code') ;

会奏效。

这不是:

INSERT INTO public.submission (id, code) VALUES (NULL, 'Some code') ;

一旦正确配置,我猜SQLAlchemy应该足够智能来生成正确的SQL INSERT语句。

参考:

答案 1 :(得分:1)

最终,我发现了什么问题,这绝对是我的错。我用来将旧数据加载到数据库(pgloader)的过程不仅仅是加载数据 - 它以某种方式覆盖了部分表定义!我能够pg_dump数据输出,重置表,然后重新加载 - 一切都按预期工作。谢谢你的理智检查!