/** * Copyright (C) 2015 MongoDB Inc. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License, version 3, * as published by the Free Software Foundation. * * 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 * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General 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 GNU Affero General 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/bson/util/builder.h" #include "mongo/db/jsobj.h" #include "mongo/db/json.h" #include "mongo/rpc/command_reply.h" #include "mongo/rpc/command_reply_builder.h" #include "mongo/rpc/get_status_from_command_result.h" #include "mongo/rpc/legacy_reply.h" #include "mongo/rpc/legacy_reply_builder.h" #include "mongo/rpc/op_msg_rpc_impls.h" #include "mongo/unittest/death_test.h" #include "mongo/unittest/unittest.h" namespace { using namespace mongo; template void testRoundTrip(rpc::ReplyBuilderInterface& replyBuilder, bool unifiedBodyAndMetadata); template void testErrors(rpc::ReplyBuilderInterface& replyBuilder); TEST(LegacyReplyBuilder, RoundTrip) { rpc::LegacyReplyBuilder r; testRoundTrip(r, true); } TEST(CommandReplyBuilder, RoundTrip) { rpc::CommandReplyBuilder r; testRoundTrip(r, false); } TEST(OpMsgReplyBuilder, RoundTrip) { rpc::OpMsgReplyBuilder r; testRoundTrip(r, true); } template void testErrors(rpc::ReplyBuilderInterface& replyBuilder); TEST(LegacyReplyBuilder, Errors) { rpc::LegacyReplyBuilder r; testErrors(r); } TEST(CommandReplyBuilder, Errors) { rpc::CommandReplyBuilder r; testErrors(r); } TEST(OpMsgReplyBuilder, Errors) { rpc::OpMsgReplyBuilder r; testErrors(r); } BSONObj buildMetadata() { BSONObjBuilder metadataTop; { BSONObjBuilder metadataGle(metadataTop.subobjStart("$gleStats")); metadataGle.append("lastOpTime", Timestamp()); metadataGle.append("electionId", OID("5592bee00d21e3aa796e185e")); } // For now we don't need a real $clusterTime and just ensure that it just round trips whatever // is there. If that ever changes, we will need to construct a real $clusterTime here. metadataTop.append("$clusterTime", BSON("bogus" << true)); return metadataTop.obj(); } BSONObj buildEmptyCommand() { const char text[] = "{ ok: 1.0, cursor: { firstBatch: [] } }"; mongo::BSONObj obj = mongo::fromjson(text); return obj; } BSONObj buildCommand() { BSONObjBuilder commandReplyBob{}; commandReplyBob.append("ok", 1.0); BSONObjBuilder cursorBuilder; BSONArrayBuilder a(cursorBuilder.subarrayStart("firstBatch")); a.append(BSON("Foo" << "Bar")); a.done(); cursorBuilder.appendIntOrLL("id", 1); cursorBuilder.append("ns", "test.$cmd.blah"); commandReplyBob.append("cursor", cursorBuilder.done()); return commandReplyBob.obj(); } BSONObj buildErrReply(const Status status, const BSONObj& extraInfo = {}) { BSONObjBuilder bob; bob.appendElements(extraInfo); bob.append("ok", 0.0); bob.append("errmsg", status.reason()); bob.append("code", status.code()); bob.append("codeName", ErrorCodes::errorString(status.code())); return bob.obj(); } TEST(CommandReplyBuilder, CommandError) { const Status status(ErrorCodes::InvalidLength, "Response payload too long"); BSONObj metadata = buildMetadata(); rpc::CommandReplyBuilder replyBuilder; replyBuilder.setCommandReply(status); replyBuilder.setMetadata(metadata); auto msg = replyBuilder.done(); rpc::CommandReply parsed(&msg); ASSERT_BSONOBJ_EQ(parsed.getMetadata(), metadata); ASSERT_BSONOBJ_EQ(parsed.getCommandReply(), buildErrReply(status)); } TEST(LegacyReplyBuilder, CommandError) { const Status status(ErrorCodes::InvalidLength, "Response payload too long"); BSONObj metadata = buildMetadata(); BSONObjBuilder extra; extra.append("a", "b"); extra.append("c", "d"); const BSONObj extraObj = extra.obj(); rpc::LegacyReplyBuilder replyBuilder; replyBuilder.setCommandReply(status, extraObj); replyBuilder.setMetadata(metadata); auto msg = replyBuilder.done(); rpc::LegacyReply parsed(&msg); const auto body = ([&] { BSONObjBuilder unifiedBuilder(buildErrReply(status, extraObj)); unifiedBuilder.appendElements(metadata); return unifiedBuilder.obj(); }()); ASSERT_BSONOBJ_EQ(parsed.getMetadata(), body); ASSERT_BSONOBJ_EQ(parsed.getCommandReply(), body); } TEST(OpMsgReplyBuilder, CommandError) { const Status status(ErrorCodes::InvalidLength, "Response payload too long"); BSONObj metadata = buildMetadata(); BSONObjBuilder extra; extra.append("a", "b"); extra.append("c", "d"); const BSONObj extraObj = extra.obj(); rpc::OpMsgReplyBuilder replyBuilder; replyBuilder.setCommandReply(status, extraObj); replyBuilder.setMetadata(metadata); auto msg = replyBuilder.done(); rpc::OpMsgReply parsed(&msg); const auto body = ([&] { BSONObjBuilder unifiedBuilder(buildErrReply(status, extraObj)); unifiedBuilder.appendElements(metadata); return unifiedBuilder.obj(); }()); ASSERT_BSONOBJ_EQ(parsed.getMetadata(), body); ASSERT_BSONOBJ_EQ(parsed.getCommandReply(), body); } template void testRoundTrip(rpc::ReplyBuilderInterface& replyBuilder, bool unifiedBodyAndMetadata) { auto metadata = buildMetadata(); auto commandReply = buildEmptyCommand(); replyBuilder.setCommandReply(commandReply); replyBuilder.setMetadata(metadata); auto msg = replyBuilder.done(); T parsed(&msg); if (unifiedBodyAndMetadata) { const auto body = ([&] { BSONObjBuilder unifiedBuilder(std::move(commandReply)); unifiedBuilder.appendElements(metadata); return unifiedBuilder.obj(); }()); ASSERT_BSONOBJ_EQ(parsed.getCommandReply(), body); ASSERT_BSONOBJ_EQ(parsed.getMetadata(), body); } else { ASSERT_BSONOBJ_EQ(parsed.getCommandReply(), commandReply); ASSERT_BSONOBJ_EQ(parsed.getMetadata(), metadata); } } template void testErrors(rpc::ReplyBuilderInterface& replyBuilder) { ErrorExtraInfoExample::EnableParserForTest whenInScope; const auto status = Status(ErrorExtraInfoExample(123), "Why does this keep failing!"); replyBuilder.setCommandReply(status); replyBuilder.setMetadata(buildMetadata()); const auto msg = replyBuilder.done(); T parsed(&msg); const Status result = getStatusFromCommandResult(parsed.getCommandReply()); ASSERT_EQ(result, status.code()); ASSERT_EQ(result.reason(), status.reason()); ASSERT(result.extraInfo()); ASSERT(result.extraInfo()); ASSERT_EQ(result.extraInfo()->data, 123); } } // namespace