/** * Tests that the cluster is balanced in parallel in one balancer round (standalone). */ (function() { 'use strict'; load("jstests/sharding/libs/find_chunks_util.js"); const st = new ShardingTest({shards: 4, other: {chunkSize: 1}}); var config = st.s0.getDB('config'); assert.commandWorked(st.s0.adminCommand({enableSharding: 'TestDB'})); st.ensurePrimaryShard('TestDB', st.shard0.shardName); function prepareCollectionForBalance(collName) { assert.commandWorked(st.s0.adminCommand({shardCollection: collName, key: {Key: 1}})); var coll = st.s0.getCollection(collName); const bigString = 'X'.repeat(1024 * 1024); // 1MB // Create 6 chunks initially and ensure they get balanced within 1 balancer round assert.commandWorked(coll.insert({Key: 1, Value: 'Test value 1', s: bigString})); assert.commandWorked(coll.insert({Key: 10, Value: 'Test value 10', s: bigString})); assert.commandWorked(coll.insert({Key: 20, Value: 'Test value 20', s: bigString})); assert.commandWorked(coll.insert({Key: 30, Value: 'Test value 30', s: bigString})); assert.commandWorked(coll.insert({Key: 40, Value: 'Test value 40', s: bigString})); assert.commandWorked(coll.insert({Key: 50, Value: 'Test value 50', s: bigString})); assert.commandWorked(st.splitAt(collName, {Key: 10})); assert.commandWorked(st.splitAt(collName, {Key: 20})); assert.commandWorked(st.splitAt(collName, {Key: 30})); assert.commandWorked(st.splitAt(collName, {Key: 40})); assert.commandWorked(st.splitAt(collName, {Key: 50})); // Move 3 of the chunks to st.shard1.shardName so we have option to do parallel balancing assert.commandWorked(st.moveChunk(collName, {Key: 30}, st.shard1.shardName)); assert.commandWorked(st.moveChunk(collName, {Key: 40}, st.shard1.shardName)); assert.commandWorked(st.moveChunk(collName, {Key: 50}, st.shard1.shardName)); assert.eq( 3, findChunksUtil.findChunksByNs(config, collName, {shard: st.shard0.shardName}).itcount()); assert.eq( 3, findChunksUtil.findChunksByNs(config, collName, {shard: st.shard1.shardName}).itcount()); } function checkCollectionBalanced(collName) { st.verifyCollectionIsBalanced(st.s.getCollection(collName)); } function countMoves(collName) { return config.changelog.find({what: 'moveChunk.start', ns: collName}).itcount(); } prepareCollectionForBalance('TestDB.TestColl1'); prepareCollectionForBalance('TestDB.TestColl2'); // Count the moveChunk start attempts accurately and ensure that only the correct number of // migrations are scheduled const testColl1InitialMoves = countMoves('TestDB.TestColl1'); const testColl2InitialMoves = countMoves('TestDB.TestColl2'); st.startBalancer(); st.awaitBalance("TestColl1", "TestDB"); st.awaitBalance("TestColl2", "TestDB"); st.stopBalancer(); checkCollectionBalanced('TestDB.TestColl1'); checkCollectionBalanced('TestDB.TestColl2'); assert.eq(2, countMoves('TestDB.TestColl1') - testColl1InitialMoves); assert.eq(2, countMoves('TestDB.TestColl2') - testColl2InitialMoves); // Ensure there are no migration errors reported assert.eq(0, config.changelog.find({what: 'moveChunk.error'}).itcount()); st.stop(); })();