_parsePath = function() { var dbpath = ""; for( var i = 0; i < arguments.length; ++i ) if ( arguments[ i ] == "--dbpath" ) dbpath = arguments[ i + 1 ]; if ( dbpath == "" ) throw "No dbpath specified"; return dbpath; } _parsePort = function() { var port = ""; for( var i = 0; i < arguments.length; ++i ) if ( arguments[ i ] == "--port" ) port = arguments[ i + 1 ]; if ( port == "" ) throw "No port specified"; return port; } createMongoArgs = function( binaryName , args ){ var fullArgs = [ binaryName ]; if ( args.length == 1 && isObject( args[0] ) ){ var o = args[0]; for ( var k in o ){ if ( k == "v" && isNumber( o[k] ) ){ var n = o[k]; if ( n > 0 ){ if ( n > 10 ) n = 10; var temp = "-"; while ( n-- > 0 ) temp += "v"; fullArgs.push( temp ); } } else { fullArgs.push( "--" + k ); if ( o[k] != "" ) fullArgs.push( "" + o[k] ); } } } else { for ( var i=0; i> " + tojson( z.max ) + " on : " + z.shard + " " + tojson( z.lastmod ) ); } ); } ); print( raw ); } ShardingTest.prototype.sync = function(){ this.adminCommand( "connpoolsync" ); } ShardingTest.prototype.onNumShards = function( collName , dbName ){ this.sync(); // we should sync since we're going directly to mongod here dbName = dbName || "test"; var num=0; for ( var i=0; i 0 ) num++; return num; } ShardingTest.prototype.shardGo = function( collName , key , split , move , dbName ){ split = split || key; move = move || split; dbName = dbName || "test"; var c = dbName + "." + collName; s.adminCommand( { shardcollection : c , key : key } ); s.adminCommand( { split : c , middle : split } ); s.adminCommand( { movechunk : c , find : move , to : this.getOther( s.getServer( dbName ) ).name } ); } MongodRunner = function( port, dbpath, peer, arbiter, extraArgs ) { this.port_ = port; this.dbpath_ = dbpath; this.peer_ = peer; this.arbiter_ = arbiter; this.extraArgs_ = extraArgs; } MongodRunner.prototype.start = function( reuseData ) { var args = []; if ( reuseData ) { args.push( "mongod" ); } args.push( "--port" ); args.push( this.port_ ); args.push( "--dbpath" ); args.push( this.dbpath_ ); if ( this.peer_ && this.arbiter_ ) { args.push( "--pairwith" ); args.push( this.peer_ ); args.push( "--arbiter" ); args.push( this.arbiter_ ); args.push( "--oplogSize" ); // small oplog by default so startup fast args.push( "1" ); } args.push( "--nohttpinterface" ); args.push( "--noprealloc" ); args.push( "--smallfiles" ); args.push( "--bind_ip" ); args.push( "127.0.0.1" ); if ( this.extraArgs_ ) { args = args.concat( this.extraArgs_ ); } removeFile( this.dbpath_ + "/mongod.lock" ); if ( reuseData ) { return startMongoProgram.apply( null, args ); } else { return startMongod.apply( null, args ); } } MongodRunner.prototype.port = function() { return this.port_; } MongodRunner.prototype.toString = function() { return [ this.port_, this.dbpath_, this.peer_, this.arbiter_ ].toString(); } ReplPair = function( left, right, arbiter ) { this.left_ = left; this.leftC_ = null; this.right_ = right; this.rightC_ = null; this.arbiter_ = arbiter; this.arbiterC_ = null; this.master_ = null; this.slave_ = null; } ReplPair.prototype.start = function( reuseData ) { if ( this.arbiterC_ == null ) { this.arbiterC_ = this.arbiter_.start(); } if ( this.leftC_ == null ) { this.leftC_ = this.left_.start( reuseData ); } if ( this.rightC_ == null ) { this.rightC_ = this.right_.start( reuseData ); } } ReplPair.prototype.isMaster = function( mongo, debug ) { var im = mongo.getDB( "admin" ).runCommand( { ismaster : 1 } ); assert( im && im.ok, "command ismaster failed" ); if ( debug ) { printjson( im ); } return im.ismaster; } ReplPair.prototype.isInitialSyncComplete = function( mongo, debug ) { var isc = mongo.getDB( "admin" ).runCommand( { isinitialsynccomplete : 1 } ); assert( isc && isc.ok, "command isinitialsynccomplete failed" ); if ( debug ) { printjson( isc ); } return isc.initialsynccomplete; } ReplPair.prototype.checkSteadyState = function( state, expectedMasterHost, twoMasterOk, leftValues, rightValues, debug ) { leftValues = leftValues || {}; rightValues = rightValues || {}; var lm = null; var lisc = null; if ( this.leftC_ != null ) { lm = this.isMaster( this.leftC_, debug ); leftValues[ lm ] = true; lisc = this.isInitialSyncComplete( this.leftC_, debug ); } var rm = null; var risc = null; if ( this.rightC_ != null ) { rm = this.isMaster( this.rightC_, debug ); rightValues[ rm ] = true; risc = this.isInitialSyncComplete( this.rightC_, debug ); } var stateSet = {} state.forEach( function( i ) { stateSet[ i ] = true; } ); if ( !( 1 in stateSet ) || ( ( risc || risc == null ) && ( lisc || lisc == null ) ) ) { if ( rm == 1 && lm != 1 ) { assert( twoMasterOk || !( 1 in leftValues ) ); this.master_ = this.rightC_; this.slave_ = this.leftC_; } else if ( lm == 1 && rm != 1 ) { assert( twoMasterOk || !( 1 in rightValues ) ); this.master_ = this.leftC_; this.slave_ = this.rightC_; } if ( !twoMasterOk ) { assert( lm != 1 || rm != 1, "two masters" ); } // check for expected state if ( state.sort().toString() == [ lm, rm ].sort().toString() ) { if ( expectedMasterHost != null ) { if( expectedMasterHost == this.master_.host ) { return true; } } else { return true; } } } this.master_ = null; this.slave_ = null; return false; } ReplPair.prototype.waitForSteadyState = function( state, expectedMasterHost, twoMasterOk, debug ) { state = state || [ 1, 0 ]; twoMasterOk = twoMasterOk || false; var rp = this; var leftValues = {}; var rightValues = {}; assert.soon( function() { return rp.checkSteadyState( state, expectedMasterHost, twoMasterOk, leftValues, rightValues, debug ); }, "rp (" + rp + ") failed to reach expected steady state (" + state + ")" ); } ReplPair.prototype.master = function() { return this.master_; } ReplPair.prototype.slave = function() { return this.slave_; } ReplPair.prototype.right = function() { return this.rightC_; } ReplPair.prototype.left = function() { return this.leftC_; } ReplPair.prototype.arbiter = function() { return this.arbiterC_; } ReplPair.prototype.killNode = function( mongo, signal ) { signal = signal || 15; if ( this.leftC_ != null && this.leftC_.host == mongo.host ) { stopMongod( this.left_.port_ ); this.leftC_ = null; } if ( this.rightC_ != null && this.rightC_.host == mongo.host ) { stopMongod( this.right_.port_ ); this.rightC_ = null; } if ( this.arbiterC_ != null && this.arbiterC_.host == mongo.host ) { stopMongod( this.arbiter_.port_ ); this.arbiterC_ = null; } } ReplPair.prototype._annotatedNode = function( mongo ) { var ret = ""; if ( mongo != null ) { ret += " (connected)"; if ( this.master_ != null && mongo.host == this.master_.host ) { ret += "(master)"; } if ( this.slave_ != null && mongo.host == this.slave_.host ) { ret += "(slave)"; } } return ret; } ReplPair.prototype.toString = function() { var ret = ""; ret += "left: " + this.left_; ret += " " + this._annotatedNode( this.leftC_ ); ret += " right: " + this.right_; ret += " " + this._annotatedNode( this.rightC_ ); return ret; } ToolTest = function( name ){ this.name = name; this.port = allocatePorts(1)[0]; this.baseName = "jstests_tool_" + name; this.root = "/data/db/" + this.baseName; this.dbpath = this.root + "/"; this.ext = this.root + "_external/"; this.extFile = this.root + "_external/a"; resetDbpath( this.dbpath ); } ToolTest.prototype.startDB = function( coll ){ assert( ! this.m , "db already running" ); this.m = startMongoProgram( "mongod" , "--port", this.port , "--dbpath" , this.dbpath , "--nohttpinterface", "--noprealloc" , "--smallfiles" , "--bind_ip", "127.0.0.1" ); this.db = this.m.getDB( this.baseName ); if ( coll ) return this.db.getCollection( coll ); return this.db; } ToolTest.prototype.stop = function(){ if ( ! this.m ) return; stopMongod( this.port ); this.m = null; this.db = null; } ToolTest.prototype.runTool = function(){ var a = [ "mongo" + arguments[0] ]; var hasdbpath = false; for ( var i=1; i