/** * 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. */ #define MONGO_LOG_DEFAULT_COMPONENT ::mongo::logger::LogComponent::kSharding #include "mongo/platform/basic.h" #include "mongo/s/chunk.h" #include "mongo/platform/random.h" #include "mongo/s/grid.h" #include "mongo/util/mongoutils/str.h" namespace mongo { namespace { // Test whether we should split once data * kSplitCheckInterval > chunkSize (approximately) PseudoRandom prng(static_cast(time(0))); // Assume user has 64MB chunkSize setting. It is ok if this assumption is wrong since it is only // a heuristic. const int64_t kMaxDataWritten = 64 / Chunk::kSplitTestFactor; /** * Generates a random value for _dataWritten so that a mongos restart wouldn't cause delay in * splitting. */ int64_t mkDataWritten() { return prng.nextInt64(kMaxDataWritten); } } // namespace Chunk::Chunk(const ChunkType& from) : _range(from.getMin(), from.getMax()), _shardId(from.getShard()), _lastmod(from.getVersion()), _jumbo(from.getJumbo()), _dataWritten(mkDataWritten()) { invariantOK(from.validate()); } bool Chunk::containsKey(const BSONObj& shardKey) const { return getMin().woCompare(shardKey) <= 0 && shardKey.woCompare(getMax()) < 0; } uint64_t Chunk::getBytesWritten() const { return _dataWritten; } uint64_t Chunk::addBytesWritten(uint64_t bytesWrittenIncrement) { _dataWritten += bytesWrittenIncrement; return _dataWritten; } void Chunk::clearBytesWritten() { _dataWritten = 0; } bool Chunk::shouldSplit(uint64_t desiredChunkSize, bool minIsInf, bool maxIsInf) const { // If this chunk is at either end of the range, trigger auto-split at 10% less data written in // order to trigger the top-chunk optimization. const uint64_t splitThreshold = (minIsInf || maxIsInf) ? static_cast((double)desiredChunkSize * 0.9) : desiredChunkSize; // Check if there are enough estimated bytes written to warrant a split return _dataWritten >= splitThreshold / kSplitTestFactor; } std::string Chunk::toString() const { return str::stream() << ChunkType::shard() << ": " << _shardId << ", " << ChunkType::lastmod() << ": " << _lastmod.toString() << ", " << _range.toString(); } void Chunk::markAsJumbo() { _jumbo = true; } } // namespace mongo