Files
mongo/jstests/aggregation/sources/out/mode_insert_documents.js
Kyle Suarez b46de3f6c0 SERVER-36100 generate _id for $out uniqueKey if not present
If the _id field is part of the $out "uniqueKey" but is not present in
the document at the time of write, the $out stage will generate it
automatically.

We won't generate an _id if if isn't part of the "uniqueKey" so that
DocumentSourceOutInPlaceReplace won't use it as part of the update.
2018-08-30 10:45:29 -04:00

137 lines
4.8 KiB
JavaScript

// Tests the behavior of $out with mode "insertDocuments".
// @tags: [assumes_unsharded_collection, assumes_no_implicit_collection_creation_after_drop]
(function() {
"use strict";
load("jstests/aggregation/extras/utils.js"); // For assertErrorCode.
load("jstests/libs/fixture_helpers.js"); // For FixtureHelpers.isMongos.
const coll = db.mode_insert_documents;
coll.drop();
const targetColl = db.mode_insert_documents_out;
targetColl.drop();
const pipeline = [{$out: {to: targetColl.getName(), mode: "insertDocuments"}}];
//
// Test $out with a non-existent output collection.
//
assert.commandWorked(coll.insert({_id: 0}));
coll.aggregate(pipeline);
assert.eq(1, targetColl.find().itcount());
//
// Test $out with an existing output collection.
//
assert.commandWorked(coll.remove({_id: 0}));
assert.commandWorked(coll.insert({_id: 1}));
coll.aggregate(pipeline);
assert.eq(2, targetColl.find().itcount());
//
// Test that $out fails if there's a duplicate key error.
//
assertErrorCode(coll, pipeline, ErrorCodes.DuplicateKey);
//
// Test that $out will preserve the indexes and options of the output collection.
//
const validator = {a: {$gt: 0}};
targetColl.drop();
assert.commandWorked(db.createCollection(targetColl.getName(), {validator: validator}));
assert.commandWorked(targetColl.createIndex({a: 1}));
coll.drop();
assert.commandWorked(coll.insert({a: 1}));
coll.aggregate(pipeline);
assert.eq(1, targetColl.find().itcount());
assert.eq(2, targetColl.getIndexes().length);
const listColl = db.runCommand({listCollections: 1, filter: {name: targetColl.getName()}});
assert.commandWorked(listColl);
assert.eq(validator, listColl.cursor.firstBatch[0].options["validator"]);
//
// Test that $out fails if it violates a unique index constraint.
//
coll.drop();
assert.commandWorked(coll.insert([{_id: 0, a: 0}, {_id: 1, a: 0}]));
targetColl.drop();
assert.commandWorked(targetColl.createIndex({a: 1}, {unique: true}));
assertErrorCode(coll, pipeline, ErrorCodes.DuplicateKey);
//
// Test that an $out aggregation succeeds even if the _id is stripped out and the "uniqueKey"
// is the document key, which will be _id for a new collection.
//
coll.drop();
assert.commandWorked(coll.insert({a: 0}));
targetColl.drop();
assert.doesNotThrow(() => coll.aggregate([
{$project: {_id: 0}},
{$out: {to: targetColl.getName(), mode: "insertDocuments"}},
]));
assert.eq(1, targetColl.find().itcount());
//
// Test that an $out aggregation succeeds even if the _id is stripped out and _id is part of a
// multi-field "uniqueKey".
//
coll.drop();
assert.commandWorked(coll.insert([{_id: "should be projected away", name: "kyle"}]));
targetColl.drop();
assert.commandWorked(targetColl.createIndex({_id: 1, name: -1}, {unique: true}));
assert.doesNotThrow(() => coll.aggregate([
{$project: {_id: 0}},
{$out: {to: targetColl.getName(), mode: "insertDocuments", uniqueKey: {_id: 1, name: 1}}},
]));
assert.eq(1, targetColl.find().itcount());
//
// Tests for $out to a database that differs from the aggregation database.
//
const foreignDb = db.getSiblingDB("mode_insert_documents_foreign");
const foreignTargetColl = foreignDb.mode_insert_documents_out;
const pipelineDifferentOutputDb = [
{$project: {_id: 0}},
{
$out: {
to: foreignTargetColl.getName(),
db: foreignDb.getName(),
mode: "insertDocuments",
}
}
];
foreignDb.dropDatabase();
coll.drop();
assert.commandWorked(coll.insert({a: 1}));
if (!FixtureHelpers.isMongos(db)) {
//
// Test that $out implicitly creates a new database when the output collection's database
// doesn't exist.
//
coll.aggregate(pipelineDifferentOutputDb);
assert.eq(foreignTargetColl.find().itcount(), 1);
} else {
// Implicit database creation is prohibited in a cluster.
const error = assert.throws(() => coll.aggregate(pipelineDifferentOutputDb));
assert.commandFailedWithCode(error, ErrorCodes.NamespaceNotFound);
// Explicitly create the collection and database, then fall through to the test below.
assert.commandWorked(foreignTargetColl.insert({val: "forcing database creation"}));
}
//
// Re-run the $out aggregation, which should merge with the existing contents of the
// collection. We rely on implicit _id generation to give us unique _id values.
//
assert.doesNotThrow(() => coll.aggregate(pipelineDifferentOutputDb));
assert.eq(foreignTargetColl.find().itcount(), 2);
}());