/** * Copyright (C) 2018-present MongoDB, Inc. * * This program is free software: you can redistribute it and/or modify * it under the terms of the Server Side Public License, version 1, * as published by MongoDB, Inc. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * Server Side Public License for more details. * * You should have received a copy of the Server Side Public License * along with this program. If not, see * . * * As a special exception, the copyright holders give permission to link the * code of portions of this program with the OpenSSL library under certain * conditions as described in each individual source file and distribute * linked combinations including the program with the OpenSSL library. You * must comply with the Server Side Public License in all respects for * all of the code used other than as permitted herein. If you modify file(s) * with this exception, you may extend this exception to your version of the * file(s), but you are not obligated to do so. If you do not wish to do so, * delete this exception statement from your version. If you delete this * exception statement from all source files in the program, then also delete * it in the license file. */ #include "mongo/platform/basic.h" #include "mongo/db/update/push_node.h" #include #include "mongo/base/simple_string_data_comparator.h" #include "mongo/bson/mutable/algorithm.h" #include "mongo/db/matcher/expression_parser.h" #include "mongo/db/update/update_internal_node.h" namespace mongo { const StringData PushNode::kEachClauseName = "$each"_sd; const StringData PushNode::kSliceClauseName = "$slice"; const StringData PushNode::kSortClauseName = "$sort"; const StringData PushNode::kPositionClauseName = "$position"; namespace { /** * When the $sort clause in a $push modifer is an object, that object should pass the checks in * this function. */ Status checkSortClause(const BSONObj& sortObject) { if (sortObject.isEmpty()) { return Status(ErrorCodes::BadValue, "The $sort pattern is empty when it should be a set of fields."); } for (auto&& patternElement : sortObject) { double orderVal = patternElement.isNumber() ? patternElement.Number() : 0; if (orderVal != -1 && orderVal != 1) { return Status(ErrorCodes::BadValue, "The $sort element value must be either 1 or -1"); } FieldRef sortField(patternElement.fieldName()); if (sortField.numParts() == 0) { return Status(ErrorCodes::BadValue, "The $sort field cannot be empty"); } for (size_t i = 0; i < sortField.numParts(); ++i) { if (sortField.getPart(i).size() == 0) { return Status(ErrorCodes::BadValue, str::stream() << "The $sort field is a dotted field " "but has an empty part: " << sortField.dottedField()); } } } return Status::OK(); } } // namespace Status PushNode::init(BSONElement modExpr, const boost::intrusive_ptr& expCtx) { invariant(modExpr.ok()); if (modExpr.type() == BSONType::Object && modExpr[kEachClauseName]) { std::set validClauseNames{ kEachClauseName, kSliceClauseName, kSortClauseName, kPositionClauseName}; auto clausesFound = SimpleStringDataComparator::kInstance.makeStringDataUnorderedMap(); for (auto&& modifier : modExpr.embeddedObject()) { auto clauseName = modifier.fieldNameStringData(); auto foundClauseName = validClauseNames.find(clauseName); if (foundClauseName == validClauseNames.end()) { return Status(ErrorCodes::BadValue, str::stream() << "Unrecognized clause in $push: " << modifier.fieldNameStringData()); } if (clausesFound.find(*foundClauseName) != clausesFound.end()) { return Status(ErrorCodes::BadValue, str::stream() << "Only one " << clauseName << " is supported."); } clausesFound.insert(std::make_pair(*foundClauseName, modifier)); } // Parse $each. auto eachIt = clausesFound.find(kEachClauseName); invariant(eachIt != clausesFound.end()); // We already checked for a $each clause. const auto& eachClause = eachIt->second; if (eachClause.type() != BSONType::Array) { return Status(ErrorCodes::BadValue, str::stream() << "The argument to $each in $push must be" " an array but it was of type: " << typeName(eachClause.type())); } for (auto&& item : eachClause.embeddedObject()) { _valuesToPush.push_back(item); } // Parse (optional) $slice. auto sliceIt = clausesFound.find(kSliceClauseName); if (sliceIt != clausesFound.end()) { auto sliceClause = sliceIt->second; auto parsedSliceValue = MatchExpressionParser::parseIntegerElementToLong(sliceClause); if (parsedSliceValue.isOK()) { _slice = parsedSliceValue.getValue(); } else { return Status(ErrorCodes::BadValue, str::stream() << "The value for $slice must " "be an integer value but was given type: " << typeName(sliceClause.type())); } } // Parse (optional) $sort. auto sortIt = clausesFound.find(kSortClauseName); if (sortIt != clausesFound.end()) { auto sortClause = sortIt->second; if (sortClause.type() == BSONType::Object) { auto status = checkSortClause(sortClause.embeddedObject()); if (status.isOK()) { _sort = PatternElementCmp(sortClause.embeddedObject(), expCtx->getCollator()); } else { return status; } } else if (sortClause.isNumber()) { double orderVal = sortClause.Number(); if (orderVal == -1 || orderVal == 1) { _sort = PatternElementCmp(BSON("" << orderVal), expCtx->getCollator()); } else { return Status(ErrorCodes::BadValue, "The $sort element value must be either 1 or -1"); } } else { return Status(ErrorCodes::BadValue, "The $sort is invalid: use 1/-1 to sort the whole element, " "or {field:1/-1} to sort embedded fields"); } } // Parse (optional) $position. auto positionIt = clausesFound.find(kPositionClauseName); if (positionIt != clausesFound.end()) { auto positionClause = positionIt->second; auto parsedPositionValue = MatchExpressionParser::parseIntegerElementToLong(positionClause); if (parsedPositionValue.isOK()) { _position = parsedPositionValue.getValue(); } else { return Status(ErrorCodes::BadValue, str::stream() << "The value for $position must " "be an integer value, not of type: " << typeName(positionClause.type())); } } } else { // No $each clause. We treat the value of $push as the element to add to the array. _valuesToPush.push_back(modExpr); } return Status::OK(); } ModifierNode::ModifyResult PushNode::insertElementsWithPosition( mutablebson::Element* array, long long position, const std::vector& valuesToPush) { if (valuesToPush.empty()) { return ModifyResult::kNoOp; } auto& document = array->getDocument(); auto firstElementToInsert = document.makeElementWithNewFieldName(StringData(), valuesToPush.front()); // We assume that no array has more than std::numerical_limits::max() elements. long long arraySize = static_cast(countChildren(*array)); // We insert the first element of 'valuesToPush' at the location requested in the 'position' // variable. ModifyResult result; if (arraySize == 0) { invariantOK(array->pushBack(firstElementToInsert)); result = ModifyResult::kNormalUpdate; } else if (position > arraySize) { invariantOK(array->pushBack(firstElementToInsert)); result = ModifyResult::kArrayAppendUpdate; } else if (position > 0) { auto insertAfter = getNthChild(*array, position - 1); invariantOK(insertAfter.addSiblingRight(firstElementToInsert)); result = ModifyResult::kNormalUpdate; } else if (position < 0 && -position < arraySize) { auto insertAfter = getNthChild(*array, arraySize - (-position) - 1); invariantOK(insertAfter.addSiblingRight(firstElementToInsert)); result = ModifyResult::kNormalUpdate; } else { invariantOK(array->pushFront(firstElementToInsert)); result = ModifyResult::kNormalUpdate; } // We insert all the rest of the elements after the one we just inserted. std::accumulate(std::next(valuesToPush.begin()), valuesToPush.end(), firstElementToInsert, [&document](auto& insertAfter, auto& valueToInsert) { auto nextElementToInsert = document.makeElementWithNewFieldName(StringData(), valueToInsert); invariantOK(insertAfter.addSiblingRight(nextElementToInsert)); return nextElementToInsert; }); return result; } ModifierNode::ModifyResult PushNode::performPush(mutablebson::Element* element, FieldRef* elementPath) const { if (element->getType() != BSONType::Array) { invariant(elementPath); // We can only hit this error if we are updating an existing path. auto idElem = mutablebson::findFirstChildNamed(element->getDocument().root(), "_id"); uasserted(ErrorCodes::BadValue, str::stream() << "The field '" << elementPath->dottedField() << "'" << " must be an array but is of type " << typeName(element->getType()) << " in document {" << (idElem.ok() ? idElem.toString() : "no id") << "}"); } auto result = insertElementsWithPosition(element, _position, _valuesToPush); if (_sort) { result = ModifyResult::kNormalUpdate; sortChildren(*element, *_sort); } // std::abs(LLONG_MIN) results in undefined behavior on 2's complement systems because the // absolute value of LLONG_MIN cannot be represented in a 'long long'. const auto sliceAbs = _slice == std::numeric_limits::min() ? std::abs(_slice + 1) : std::abs(_slice); while (static_cast(countChildren(*element)) > sliceAbs) { result = ModifyResult::kNormalUpdate; if (_slice >= 0) { invariantOK(element->popBack()); } else { // A negative value in '_slice' trims the array down to abs(_slice) but removes entries // from the front of the array instead of the back. invariantOK(element->popFront()); } } return result; } ModifierNode::ModifyResult PushNode::updateExistingElement( mutablebson::Element* element, std::shared_ptr elementPath) const { return performPush(element, elementPath.get()); } void PushNode::logUpdate(LogBuilder* logBuilder, StringData pathTaken, mutablebson::Element element, ModifyResult modifyResult) const { invariant(logBuilder); if (modifyResult == ModifyResult::kNormalUpdate || modifyResult == ModifyResult::kCreated) { // Simple case: log the entires contents of the updated array. uassertStatusOK(logBuilder->addToSetsWithNewFieldName(pathTaken, element)); } else if (modifyResult == ModifyResult::kArrayAppendUpdate) { // This update only modified the array by appending entries to the end. Rather than writing // out the entire contents of the array, we create oplog entries for the newly appended // elements. auto numAppended = _valuesToPush.size(); auto arraySize = countChildren(element); invariant(arraySize > numAppended); auto position = arraySize - numAppended; for (const auto& valueToLog : _valuesToPush) { std::string pathToArrayElement(str::stream() << pathTaken << "." << position); uassertStatusOK(logBuilder->addToSetsWithNewFieldName(pathToArrayElement, valueToLog)); ++position; } } else { MONGO_UNREACHABLE; } } void PushNode::setValueForNewElement(mutablebson::Element* element) const { BSONObj emptyArray; invariantOK(element->setValueArray(emptyArray)); (void)performPush(element, nullptr); } } // namespace mongo