5

我正在尝试使用 Node.js(使用本机 MongoDB 驱动程序)在 MongoDB 中设置和更新一些上限集合。我的目标是,在运行 app.js 时,将文档插入到上限集合中,并更新上限集合中的现有文档。这两个都在运行setInterval(),所以每隔几秒钟。

我的问题:

  1. 如果集合尚不存在,我想创建一个集合,但如果存在,我想将文档插入其中。检查这个的正确方法是什么?
  2. 对于有上限的集合,我是否应该在向其中插入任何内容之前先明确地创建它们?通常我相信你可以在不先明确创建它们的情况下将它们插入到集合中,但在这种情况下,我需要确保它们被封顶。一旦存在上限集合,我就知道如何将新文档插入其中,问题是我需要一些方法来处理第一次使用的应用程序(在新服务器上),其中集合尚不存在,我想要使用节点进行此创建,而不必跳入 mongo cli。
  3. 这里的诀窍是集合需要设置上限,所以我可以执行以下操作db.createCollection("collectionName", { capped : true, size : 100000, max : 5000 } ):这将为我创建上限集合,但每次我调用它时它都会调用createCollection()而不是更新或插入 - 如果我调用createCollection(),一旦集合已经存在,它会完全覆盖现有集合吗?
  4. 另一种方法是将集合转换为具有上限的集合:db.runCommand({"convertToCapped": "collectionName", size: 100000, max : 5000 });. 这样做的问题是节点不认为runCommand()是一个有效的函数并且它会出错。还有什么我打算打电话来让它工作的吗?它适用于 mongo cli,但不适用于节点
  5. 您使用什么类型的查询来查找集合中的第一个文档?同样,在 mongo cli 中我可以使用db.collections.find()一些查询,但在节点中它指出这find()不是一个有效的函数
  6. 我将如何使用collection.update()向现有文档添加一些新字段?可以说文档是一些简单的对象{key1: "value", key2: "value"},例如,但我有一个包含{key3: "value"}. 当前文档中不存在密钥 3,如何将其添加到当前存在的内容中?这与上面的 #4 有点相关,因为我不确定应该传入什么作为查询参数,因为它find()似乎不适用于 node.js。
4

1 回答 1

3

关于您关于上限集合和自动创建的问题 1 - 4,有几种方法可以做到这一点。一方面,您可以运行一个脚本来初始化您的数据库,这样当您第一次运行它时,它就可以为您的客户提供可用的上限集合。另一方面,您可以在插入文档之前检查给定集合中是否有任何文档。如果有,您只需插入您的文档,如果没有,您创建 capped 集合,然后插入文档作为对该函数的回调。它会像这样工作:

var host = "localhost",
port = 27017,
dbName = "so";

var MongoClient = require('mongodb').MongoClient, Server = require('mongodb').Server;
var mongoclient = new MongoClient(new Server(host, port));
var db = mongoclient.db(dbName);

db.open(function(err, db) {

    if(err) throw err;

    // Capped collection.
    var capped = db.collection('capped');

    // Document to be inserted.
    var document =  { "foo": 1, "bar": 1 }

    capped.find().count(function(err, count) {

        if(err) throw err;

        if (count === 0) {          
            console.log("Creating collection...");
            db.createCollection("capped",
                                { "capped": true,
                                  "size": 100000,
                                  "max": 5000 },
                                function(err, collection) {
                if(err) throw err;              

                // Insert a document here.
                console.log("Inserting document...");
                collection.insert(document, function(err, result) {
                    if (err) throw err;                 
                });
            });

        } else {

            // Insert your document here without creating collection.
            console.log("Inserting document without creating collection...");
            capped.insert(document, function(err, result) {
                if (err) throw err;             
            });
        }

    });
});

关于问题 5,您可以使用findOne()在集合中查找文档,但这不一定是第一个或最后一个。如果你想保证第一个或最后一个,你可以运行 afind()sort()1。limit()按升序排序_id应该给你第一个文档。更多信息在这里

// Sort 1 for ascending, -1 for descending.
capped.find().sort([["_id", 1]]).limit(1).nextObject(function(err, item) {
    console.log(item);
});

最后,对于问题 6,您只需将$set运算符与update()方法一起使用。更多信息在这里

capped.update({ "foo": 1 }, { "$set": { "bar": 2 } }, {}, function(err, result) {
    console.log(result);
});

请注意,您只能为有上限的集合更新文档,因此您不能插入您提到的额外字段。您可能需要注意此处列举的其他限制。

[编辑:在最后一个文档中添加更新嵌套字段。]

如果要更新第一个或最后一个文档中的嵌套字段(分别在排序中使用 1 或 -1),您可以获取文档,提取_id,然后对该文档执行原子更新。像这样的东西:

capped.find().sort([["_id", -1]]).limit(1).nextObject(function(err, item) {       
    if(err) throw err;
    capped.update({ "_id": item._id },
              { "$set": { "timeCollected": 15, "publicIP.ip" : "127.0.0.1" }},
              function(err, result) {

        if(err) throw err;
        console.log(result);    
    });
});  

请注意,即使在上限集合中更新文档中存在的字段时,您也需要确保新值适合为文档分配的空间。因此,例如,将字符串值从"1"to更新"127.0.0.1"不一定有效。

于 2015-02-24T01:33:00.120 回答