我最近开始将一个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
我还是新手,但不应该有序列吗?
任何关于下一步寻找什么的见解或建议都将受到超级赞赏。
答案 0 :(得分:3)
标准SQL是PRIMARY KEY
UNIQUE
和NOT 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数据输出,重置表,然后重新加载 - 一切都按预期工作。谢谢你的理智检查!