80

我正在跟进我之前提出的一个问题,在该问题中我试图寻求从愚蠢/写得不好的 mysql 查询到 postgresql 的转换。我相信我成功了。无论如何,我使用的是手动从 mysql 数据库移动到 postgres 数据库的数据。我正在使用如下所示的查询:

  UPDATE krypdos_coderound cru

  set is_correct = case 
      when t.kv_values1 = t.kv_values2 then True 
      else False 
      end

  from 
  
  (select cr.id, 
    array_agg(
    case when kv1.code_round_id = cr.id 
    then kv1.option_id 
    else null end 
    ) as kv_values1,

    array_agg(
    case when kv2.code_round_id = cr_m.id 
    then kv2.option_id 
    else null end 
    ) as kv_values2

    from krypdos_coderound cr
     join krypdos_value kv1 on kv1.code_round_id = cr.id
     join krypdos_coderound cr_m 
       on cr_m.object_id=cr.object_id 
       and cr_m.content_type_id =cr.content_type_id 
     join krypdos_value kv2 on kv2.code_round_id = cr_m.id

   WHERE
     cr.is_master= False
     AND cr_m.is_master= True 
     AND cr.object_id=%s 
     AND cr.content_type_id=%s 

   GROUP BY cr.id  
  ) t

where t.id = cru.id
    """ % ( self.object_id, self.content_type.id)
  )

我有理由相信这很好用。然而,这又引出了一个新问题。尝试提交时,我从 django 收到一条错误消息,指出:

IntegrityError at (some url): 
duplicate key value violates unique constraint "krypdos_value_pkey"

我查看了此处发布的一些回复,但我还没有找到解决问题的方法(尽管相关问题已经引起了一些有趣的阅读)。我在我的日志中看到了这一点,这很有趣,因为我从不明确调用 insert-django 必须处理它:

   STATEMENT:  INSERT INTO "krypdos_value" ("code_round_id", "variable_id", "option_id", "confidence", "freetext")
   VALUES (1105935, 11, 55, NULL, E'') 
   RETURNING "krypdos_value"."id"

但是,尝试运行会导致重复键错误。实际错误在下面的代码中抛出。

 # Delete current coding         CodeRound.objects.filter(object_id=o.id,content_type=object_type,is_master=True).delete()
  code_round = CodeRound(object_id=o.id,content_type=object_type,coded_by=request.user,comments=request.POST.get('_comments',None),is_master=True)
  code_round.save()
  for key in request.POST.keys():
    if key[0] != '_' or key != 'csrfmiddlewaretoken':
      options = request.POST.getlist(key)
      for option in options:
        Value(code_round=code_round,variable_id=key,option_id=option,confidence=request.POST.get('_confidence_'+key, None)).save()  #This is where it dies
  # Resave to set is_correct
  code_round.save()
  o.status = '3' 
  o.save(

我已经检查了序列等,它们似乎是有序的。在这一点上,我不确定该怎么做——我认为这是 django 的结果,但我不确定。任何反馈将不胜感激!

4

14 回答 14

172

这发生在我身上——事实证明你需要在 Postgres 中重新同步你的主键字段。关键是SQL语句:

SELECT setval('tablename_id_seq', (SELECT MAX(id) FROM tablename)+1);
于 2012-06-19T01:41:00.280 回答
38

这似乎是 MySQL 和 SQLite 后端和其他后端(如 Postgres、Oracle 等)之间的已知行为差异(它们更新下一个可用的主键,即使在插入具有显式 id 的对象时也是如此)(它们没有) .

有一张票描述了同样的问题。即使它被关闭为无效,它也提供了一个提示,即有一个 Django 管理命令来更新下一个可用密钥。

要显示更新应用程序MyApp的所有下一个 ID 的 SQL :

python manage.py sqlsequencereset MyApp

为了执行该语句,您可以提供它作为dbshel​​l管理命令的输入。对于 bash,您可以键入:

python manage.py sqlsequencereset MyApp | python manage.py dbshell

管理命令的优点是抽象出底层数据库后端,因此即使稍后迁移到不同的后端也可以工作。

于 2016-05-12T10:18:50.520 回答
31

我的“库存”应用程序中有一个现有表,我想在 Django 管理中添加新记录,但出现此错误:

重复的键值违反了唯一约束“inventory_part_pkey”详细信息:键 (part_id)=(1) 已存在。

如前所述,我运行下面的代码来获取重置 id-s 的 SQL 命令:

python manage.py sqlsequencereset inventory

将管道连接python manage.py sqlsequencereset inventory | python manage.py dbshell到外壳不起作用

  • 所以我复制了生成的原始 SQL 命令
  • 然后为 postgreSQL 打开 pgAdmin3https://www.pgadmin.org并打开我的数据库
  • 点击 6. 图标(执行任意 SQL 查询)
  • 复制了生成的语句

在我的情况下,原始 SQL 命令是:

BEGIN;
SELECT setval(pg_get_serial_sequence('"inventory_signup"','id'), coalesce(max("id"), 1), max("id") IS NOT null) FROM "inventory_signup";
SELECT setval(pg_get_serial_sequence('"inventory_supplier"','id'), coalesce(max("id"), 1), max("id") IS NOT null) FROM "inventory_supplier";
COMMIT;

用F5执行它。

这解决了一切。

于 2017-09-22T15:45:50.353 回答
9

除了 zapphods 回答:

在我的情况下,索引确实不正确,因为我已经删除了所有迁移,并且数据库在开发时可能 10-15 次,因为我没有处于迁移任何阶段。

我收到一个 IntegrityErrorfinished_product_template_finishedproduct_pkey

重新索引表并重新启动 runserver:

我正在使用 pgadmin3 并且对于哪个索引不正确并抛出重复的键错误,我导航到constraints并重新索引。

在此处输入图像描述

然后重新索引。

在此处输入图像描述

于 2014-12-21T20:25:52.573 回答
8

解决方案是您需要重新同步您的主键字段,正如“Hacking Life”所报告的那样,他编写了一个示例 SQL 代码,但正如“Ad N”所建议的那样,最好运行 Django 命令sqlsequencereset以获取您可以使用的确切 SQL 代码复制并粘贴或使用另一个命令运行。

作为对这些答案的进一步改进,我建议您和其他读者不要复制和粘贴 SQL 代码,而是更安全地以sqlsequencereset这种方式执行从您的 python 代码中生成的 SQL 查询(使用默认数据库) :

from django.core.management.color import no_style
from django.db import connection

from myapps.models import MyModel1, MyModel2


sequence_sql = connection.ops.sequence_reset_sql(no_style(), [MyModel1, MyModel2])
with connection.cursor() as cursor:
    for sql in sequence_sql:
        cursor.execute(sql)

我用Python3.6Django 2.0PostgreSQL 10测试了这段代码。

于 2018-05-10T15:34:42.673 回答
5

如果你想像我一样重置所有表上的 PK,你可以使用PostgreSQL 推荐的方式

SELECT 'SELECT SETVAL(' ||
       quote_literal(quote_ident(PGT.schemaname) || '.' || quote_ident(S.relname)) ||
       ', COALESCE(MAX(' ||quote_ident(C.attname)|| '), 1) ) FROM ' ||
       quote_ident(PGT.schemaname)|| '.'||quote_ident(T.relname)|| ';'
FROM pg_class AS S,
     pg_depend AS D,
     pg_class AS T,
     pg_attribute AS C,
     pg_tables AS PGT
WHERE S.relkind = 'S'
    AND S.oid = D.objid
    AND D.refobjid = T.oid
    AND D.refobjid = C.attrelid
    AND D.refobjsubid = C.attnum
    AND T.relname = PGT.tablename
ORDER BY S.relname;

运行此查询后,您将需要执行查询的结果。我通常复制并粘贴到记事本中。然后我找到并替换"SELECTSELECT和。我复制并粘贴到 pgAdmin III 并运行查询。它重置数据库中的所有表。上面的链接提供了更多“专业”说明。;";

于 2018-03-14T17:56:30.450 回答
3

如果您手动复制了数据库,您可能会遇到此处描述的问题

于 2012-06-18T20:00:04.930 回答
3

我遇到了这个错误,因为我以错误的方式将额外的参数传递给了 save 方法。

对于遇到这种情况的任何人,请尝试使用以下命令强制更新:

instance_name.save(..., force_update=True)

如果您同时遇到无法传递的错误,则force_insert可能force_update像我一样以错误的方式传递了一些自定义参数。

于 2014-01-02T13:01:14.720 回答
0

我遇到了与 OP 相同的错误。

我创建了一些 Django 模型,基于模型创建了 Postgres 表,并通过 Django Admin 向 Postgres 表添加了一些行。然后我摆弄了模型中的一些列(围绕 ForeignKeys 进行更改等),但忘记了迁移更改。

运行迁移命令解决了我的问题,考虑到上面的 SQL 答案,这是有道理的。

要查看将应用哪些更改,而不实际应用它们:
python manage.py makemigrations --dry-run --verbosity 3

如果您对这些更改感到满意,请运行:
python manage.py makemigrations

然后运行:
python manage.py migrate

于 2018-12-15T01:08:51.517 回答
0

我遇到了类似的问题,但似乎没有任何效果。如果您需要数据(即在转储时不能排除它),请确保您已关闭(注释)任何 post_save 接收器。我认为数据会被导入,但由于这些,它会再次创建相同的模型。为我工作。

于 2019-11-10T09:54:39.377 回答
0

您只需要转到 pgAdmin III 并使用表的名称执行您的脚本:

SELECT setval('tablename_id_seq', (SELECT MAX(id) FROM tablename)+1);
于 2020-09-16T19:28:52.170 回答
0

根据Paolo Melchiorre 的回答,我写了一个块作为在任何 .save() 之前调用的函数

from django.db import connection
def setSqlCursor(db_table):
    sql = """SELECT pg_catalog.setval(pg_get_serial_sequence('"""+db_table+"""', 'id'), MAX(id)) FROM """+db_table+""";"""
    with connection.cursor() as cursor:
        cursor.execute(sql)
于 2021-06-22T21:36:35.187 回答
0

这是正确的说法。大多数情况下,当我们插入带有 id 字段的行时会发生这种情况。

SELECT setval('tablename_id_seq', (SELECT MAX(id) FROM tablename));
于 2021-08-02T21:45:48.723 回答
0

这个问题大约是 9 年前提出的,很多人给出了自己的解决方法。

对我来说,我输入unique=True了我的email自定义模型字段,但是在创建超级用户时我并没有要求email强制。

现在,在创建超级用户后,我的电子邮件字段只是保存为空白或Null. 现在这就是我创建和保存新用户的方式

obj = mymodel.objects.create_user(username='abc', password='abc')
obj.email = 'abc@abc.com'
obj.save()

它只是在第一行抛出了错误duplicate-key-value-violates,因为电子邮件默认设置为空,这与管理员用户相同。Django 发现了一个重复!

解决方案

  • 选项1:在创建任何用户时强制发送电子邮件(对于超级用户也是如此)
  • 选项 2:删除unique=True并运行迁移
  • 选项3:如果您不知道重复项在哪里,您可以删除该列,也可以使用清除数据库python manage.py flush

强烈建议您了解您的案例中发生错误的原因。

于 2022-01-03T10:51:04.297 回答