/** * 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/base/data_range.h" #include #include "mongo/base/data_type_endian.h" #include "mongo/platform/endian.h" #include "mongo/unittest/unittest.h" namespace mongo { namespace { // ConstDataRange::operator==() requires that the pointers // refer to the same memory addresses. // So just promote to a string that we can do direct comparisons on. std::string toString(ConstDataRange cdr) { return std::string(cdr.data(), cdr.length()); } // The ASSERT macro can't handle template specialization, // so work out the value external to the macro call. template bool isConstDataRange(const T& val) { return std::is_same_v; } template bool isDataRange(const T& val) { return std::is_same_v; } } // namespace TEST(DataRange, ConstDataRange) { unsigned char buf[sizeof(uint32_t) * 3]; uint32_t native = 1234; uint32_t le = endian::nativeToLittle(native); uint32_t be = endian::nativeToBig(native); std::memcpy(buf, &native, sizeof(uint32_t)); std::memcpy(buf + sizeof(uint32_t), &le, sizeof(uint32_t)); std::memcpy(buf + sizeof(uint32_t) * 2, &be, sizeof(uint32_t)); ConstDataRange cdv(buf, buf + sizeof(buf)); ASSERT_EQUALS(native, cdv.read()); ASSERT_EQUALS(native, cdv.read>(sizeof(uint32_t))); ASSERT_EQUALS(native, cdv.read>(sizeof(uint32_t) * 2)); auto result = cdv.readNoThrow(sizeof(uint32_t) * 3); ASSERT_EQUALS(false, result.isOK()); ASSERT_EQUALS(ErrorCodes::Overflow, result.getStatus().code()); } TEST(DataRange, ConstDataRangeType) { char buf[] = "foo"; ConstDataRange cdr(buf, buf + sizeof(buf)); ConstDataRange out(nullptr, nullptr); auto inner = cdr.readIntoNoThrow(&out); ASSERT_OK(inner); ASSERT_EQUALS(buf, out.data()); } TEST(DataRange, DataRange) { uint8_t buf[sizeof(uint32_t) * 3]; uint32_t native = 1234; DataRange dv(buf, buf + sizeof(buf)); ASSERT_OK(dv.writeNoThrow(native)); ASSERT_OK(dv.writeNoThrow(LittleEndian(native), sizeof(uint32_t))); ASSERT_OK(dv.writeNoThrow(BigEndian(native), sizeof(uint32_t) * 2)); auto result = dv.writeNoThrow(native, sizeof(uint32_t) * 3); ASSERT_NOT_OK(result); ASSERT_EQUALS(ErrorCodes::Overflow, result.code()); ASSERT_EQUALS(native, dv.read()); ASSERT_EQUALS(native, dv.read>(sizeof(uint32_t))); ASSERT_EQUALS(native, dv.read>(sizeof(uint32_t) * 2)); ASSERT_NOT_OK(dv.readNoThrow(sizeof(uint32_t) * 3)); } TEST(DataRange, DataRangeType) { char buf[] = "foo"; char buf2[] = "barZ"; DataRange dr(buf, buf + sizeof(buf) + -1); DataRange out(nullptr, nullptr); Status status = dr.readIntoNoThrow(&out); ASSERT_OK(status); ASSERT_EQUALS(buf, out.data()); dr = DataRange(buf2, buf2 + sizeof(buf2) + -1); status = dr.writeNoThrow(out); ASSERT_OK(status); ASSERT_EQUALS(std::string("fooZ"), buf2); } TEST(DataRange, InitFromContainer) { std::vector vec(20); ConstDataRange dr(vec); DataRange out(nullptr, nullptr); ASSERT_OK(dr.readIntoNoThrow(&out)); DataRange mutableDr(vec); ASSERT_OK(mutableDr.writeNoThrow(6)); std::array array; DataRange arrDR(array); auto status = arrDR.writeNoThrow(std::numeric_limits::max()); ASSERT_EQUALS(status, ErrorCodes::Overflow); } TEST(DataRange, slice) { std::string buffer("Hello World"); ConstDataRange bufferCDR(buffer.c_str(), buffer.size()); // Split by position in range [0..length) auto helloByLen = bufferCDR.slice(5); ASSERT_EQ(helloByLen.length(), 5); ASSERT_EQ(toString(helloByLen), "Hello"); // Split by pointer within range auto helloByPtr = bufferCDR.slice(bufferCDR.data() + 4); ASSERT_EQ(helloByPtr.length(), 4); ASSERT_EQ(toString(helloByPtr), "Hell"); // Get DataRange from a DataRange if original was non-const. DataRange mutableDR(const_cast(buffer.c_str()), buffer.size()); auto mutableByLen = mutableDR.slice(1); ASSERT_TRUE(isDataRange(mutableByLen)); // Get ConstDataRange from a DataRange if original was const. const DataRange nonmutableDR = mutableDR; auto nonmutableCDR = nonmutableDR.slice(2); ASSERT_TRUE(isConstDataRange(nonmutableCDR)); } TEST(DataRange, sliceThrow) { std::string buffer("Hello World"); ConstDataRange bufferCDR(buffer.c_str(), buffer.size()); // Split point is out of range. ASSERT_THROWS(bufferCDR.slice(bufferCDR.length() + 1), AssertionException); ASSERT_THROWS(bufferCDR.slice(bufferCDR.data() + bufferCDR.length() + 1), AssertionException); ASSERT_THROWS(bufferCDR.slice(bufferCDR.data() - 1), AssertionException); } TEST(DataRange, split) { std::string buffer("Hello World"); ConstDataRange bufferCDR(buffer.c_str(), buffer.size()); // Split by position in range [0..length) auto [hello, world] = bufferCDR.split(6); ASSERT_EQ(toString(hello), "Hello "); ASSERT_EQ(toString(world), "World"); // Split by pointer within range auto [hell, oWorld] = bufferCDR.split(bufferCDR.data() + 4); ASSERT_EQ(toString(hell), "Hell"); ASSERT_EQ(toString(oWorld), "o World"); // Get DataRange from a DataRange if original was non-const. DataRange bufferDR(const_cast(buffer.c_str()), buffer.size()); auto [dr1, dr2] = bufferDR.split(6); ASSERT_TRUE(isDataRange(dr1)); ASSERT_TRUE(isDataRange(dr2)); // Get ConstDataRange from a DataRange if original was const. const DataRange constBufferDR = bufferDR; auto [cdr1, cdr2] = constBufferDR.split(6); ASSERT_TRUE(isConstDataRange(cdr1)); ASSERT_TRUE(isConstDataRange(cdr2)); } TEST(DataRange, splitThrow) { std::string buffer("Hello World"); ConstDataRange bufferCDR(buffer.c_str(), buffer.size()); // Split point is out of range. ASSERT_THROWS(bufferCDR.split(bufferCDR.length() + 1), AssertionException); ASSERT_THROWS(bufferCDR.split(bufferCDR.data() + bufferCDR.length() + 1), AssertionException); ASSERT_THROWS(bufferCDR.split(bufferCDR.data() - 1), AssertionException); } } // namespace mongo