database - 为什么 Postgres 会在空表上引发唯一约束冲突?

标签 database postgresql sails.js waterline

好的,我有一个在我看来非常奇怪的问题。我有一个使用以下 SQL 创建的 Postgres 表:

CREATE TABLE message
(
  message text,
  author integer,
  thread integer,
  id serial NOT NULL,
  "createdAt" timestamp with time zone,
  "updatedAt" timestamp with time zone,
  CONSTRAINT message_pkey PRIMARY KEY (id)
)
WITH (
  OIDS=FALSE
);
  ALTER TABLE message
  OWNER TO glenselle;

我正在使用 Sails.js(它使用 Waterline ORM)将消息模型保存到数据库中。请注意,在尝试隔离问题的过程中,我每次尝试保存新记录时都开始删除表格,并且行为始终相同。 ORM 正在为我做一些关联,将作者与用户模型相关联,并将线程与线程模型相关联。无论如何,当我尝试保存记录时,我首先得到这个:

ERROR:  duplicate key value violates unique constraint "message_pkey"
DETAIL:  Key (id)=(1) already exists.
STATEMENT:  INSERT INTO "message" ("message", "author", "thread", "id", "createdAt", "updatedAt") values ($1, $2, $3, $4, $5, $6) RETURNING *

所以这应该很容易理解。表中已经有一行 id 为 1,这就是违反“message_pkey”约束的原因。但讽刺的是,没有数据!所以我的问题是,如果表中绝对没有数据(它只是被删除并使用我在上面发布的 SQL 重新创建),可能会发生什么导致 Postgres 引发唯一约束冲突?

这是我正在运行以创建模型的内容:

create: function(req, res) {
    var self = this;

    Thread.create({}, function(err, newThread) {
        if(err) return console.log(err);

        Message.create({message: req.body.conversation[0].message}, function(err, newMessage) {
            if(err) return console.log(err);
            // This returns an array of user ids
            sails.controllers.thread.parseUserIds(req.body.participants, req.user, function(idList) {

                // First, associate the message with the author
                newMessage.author = req.user.id;
                newMessage.save(function(err, savedMessage) {
                    if(err) return console.log(err);

                    // First, associate the participants with the thread
                    newThread.participants.add(idList);

                    // Next, associate the message with the thread
                    newThread.conversation.add(savedMessage);

                    newThread.save(function(err, savedThread) {
                        if(err) return console.log(err);

                        console.log('The thread looks to have been saved. Check it out!');
                        return res.json(savedThread);
                    });
                });
            });
        });
    });
},

最佳答案

您正在尝试将模型实例传递给 newThread.conversation.add,这是“创建和添加”的快捷方式。要将现有实例添加到集合中,您需要传递其 ID。将行更改为:

newThread.conversation.add(savedMessage.id);

它应该可以工作。

其他几点:

  1. 与其直接通过 sails.controllers.thread.parseUserIds 访问 Controller 方法,不如考虑将该代码移至服务(在 /api/services) 文件夹。服务由 Sails 自动全局化,并且可以从任何 Controller 访问。
  2. 如果您尝试将 ID 数组传递给 newThread.participants.add,那么最终也会失败。您需要遍历数组并为每个单独的元素调用 add

在不久的将来可能会添加对 ID(和传递现有实例对象)的支持,但我们不想让花里胡哨的东西延迟基线关联支持的发布。进行这些调整,您的代码现在应该可以工作了!

关于database - 为什么 Postgres 会在空表上引发唯一约束冲突?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22291977/

相关文章:

mysql - 数据库表设计

java - servlet 无法保存到数据库

mysql - 在 MySQL 中创建的 VIEW 使用 SELECT 给出错误的输出

postgresql - 如何对 md5 哈希值进行异或并将它们转换为 postgresql 中的 HEX

来自 postgresql 数据库的 Python 格式日期字符串

node.js - SailsJs 中的全局函数

session - sails.js session - rememberme 功能

node.js - 在 Sails js 中调用 REST API

MySQL:正向工程,错误 1826

django - API设计: can I stop users from making queries that are extremely slow?