2017-05-26 04:48:44 +08:00
/*
* SimulatedCluster . actor . cpp
*
* This source file is part of the FoundationDB open source project
*
* Copyright 2013 - 2018 Apple Inc . and the FoundationDB project authors
2018-02-22 02:25:11 +08:00
*
2017-05-26 04:48:44 +08:00
* Licensed under the Apache License , Version 2.0 ( the " License " ) ;
* you may not use this file except in compliance with the License .
* You may obtain a copy of the License at
2018-02-22 02:25:11 +08:00
*
2017-05-26 04:48:44 +08:00
* http : //www.apache.org/licenses/LICENSE-2.0
2018-02-22 02:25:11 +08:00
*
2017-05-26 04:48:44 +08:00
* Unless required by applicable law or agreed to in writing , software
* distributed under the License is distributed on an " AS IS " BASIS ,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND , either express or implied .
* See the License for the specific language governing permissions and
* limitations under the License .
*/
# include <fstream>
2020-08-07 16:02:07 +08:00
# include <ostream>
2017-05-26 04:48:44 +08:00
# include "fdbrpc/simulator.h"
# include "fdbclient/DatabaseContext.h"
2019-02-18 11:25:16 +08:00
# include "fdbserver/TesterInterface.actor.h"
2019-02-18 11:13:26 +08:00
# include "fdbserver/WorkerInterface.actor.h"
2017-05-26 04:48:44 +08:00
# include "fdbclient/ClusterInterface.h"
2018-10-20 01:30:13 +08:00
# include "fdbserver/Knobs.h"
2017-05-26 04:48:44 +08:00
# include "fdbserver/CoordinationInterface.h"
# include "fdbmonitor/SimpleIni.h"
# include "fdbrpc/AsyncFileNonDurable.actor.h"
2019-02-18 09:38:13 +08:00
# include "fdbclient/ManagementAPI.actor.h"
2019-02-18 07:41:16 +08:00
# include "fdbclient/NativeAPI.actor.h"
2019-02-18 07:19:05 +08:00
# include "fdbclient/BackupAgent.actor.h"
2020-05-21 04:23:02 +08:00
# include "fdbclient/versions.h"
2018-08-11 06:18:24 +08:00
# include "flow/actorcompiler.h" // This must be the last #include.
2017-05-26 04:48:44 +08:00
# undef max
# undef min
extern " C " int g_expect_full_pointermap ;
2019-11-16 04:26:51 +08:00
extern const char * getSourceVersion ( ) ;
2017-05-26 04:48:44 +08:00
const int MACHINE_REBOOT_TIME = 10 ;
bool destructed = false ;
template < class T >
T simulate ( const T & in ) {
BinaryWriter writer ( AssumeVersion ( currentProtocolVersion ) ) ;
writer < < in ;
BinaryReader reader ( writer . getData ( ) , writer . getLength ( ) , AssumeVersion ( currentProtocolVersion ) ) ;
T out ;
reader > > out ;
return out ;
}
ACTOR Future < Void > runBackup ( Reference < ClusterConnectionFile > connFile ) {
state std : : vector < Future < Void > > agentFutures ;
2020-09-29 04:32:20 +08:00
2020-11-23 11:49:16 +08:00
while ( g_simulator . backupAgents = = ISimulator : : BackupAgentType : : WaitForType ) {
2018-08-11 04:57:10 +08:00
wait ( delay ( 1.0 ) ) ;
2017-05-26 04:48:44 +08:00
}
2020-11-23 11:49:16 +08:00
if ( g_simulator . backupAgents = = ISimulator : : BackupAgentType : : BackupToFile ) {
2018-09-22 06:58:14 +08:00
Database cx = Database : : createDatabase ( connFile , - 1 ) ;
2017-05-26 04:48:44 +08:00
state FileBackupAgent fileAgent ;
state double backupPollDelay = 1.0 / CLIENT_KNOBS - > BACKUP_AGGREGATE_POLL_RATE ;
agentFutures . push_back ( fileAgent . run ( cx , & backupPollDelay , CLIENT_KNOBS - > SIM_BACKUP_TASKS_PER_AGENT ) ) ;
2020-11-23 11:49:16 +08:00
while ( g_simulator . backupAgents = = ISimulator : : BackupAgentType : : BackupToFile ) {
2018-08-11 04:57:10 +08:00
wait ( delay ( 1.0 ) ) ;
2017-05-26 04:48:44 +08:00
}
for ( auto it : agentFutures ) {
it . cancel ( ) ;
}
}
2018-02-21 05:22:31 +08:00
2018-08-11 08:25:43 +08:00
wait ( Future < Void > ( Never ( ) ) ) ;
2018-02-21 05:22:31 +08:00
throw internal_error ( ) ;
}
ACTOR Future < Void > runDr ( Reference < ClusterConnectionFile > connFile ) {
state std : : vector < Future < Void > > agentFutures ;
2020-11-23 11:49:16 +08:00
while ( g_simulator . drAgents = = ISimulator : : BackupAgentType : : WaitForType ) {
2018-08-11 04:57:10 +08:00
wait ( delay ( 1.0 ) ) ;
2018-02-21 05:22:31 +08:00
}
2020-11-23 11:49:16 +08:00
if ( g_simulator . drAgents = = ISimulator : : BackupAgentType : : BackupToDB ) {
2018-09-22 06:58:14 +08:00
Database cx = Database : : createDatabase ( connFile , - 1 ) ;
2017-05-26 04:48:44 +08:00
2020-11-07 15:50:55 +08:00
auto extraFile = makeReference < ClusterConnectionFile > ( * g_simulator . extraDB ) ;
2018-09-22 06:58:14 +08:00
state Database extraDB = Database : : createDatabase ( extraFile , - 1 ) ;
2017-05-26 04:48:44 +08:00
2018-06-09 02:11:08 +08:00
TraceEvent ( " StartingDrAgents " ) . detail ( " ConnFile " , connFile - > getConnectionString ( ) . toString ( ) ) . detail ( " ExtraString " , extraFile - > getConnectionString ( ) . toString ( ) ) ;
2017-05-26 04:48:44 +08:00
state DatabaseBackupAgent dbAgent = DatabaseBackupAgent ( cx ) ;
state DatabaseBackupAgent extraAgent = DatabaseBackupAgent ( extraDB ) ;
state double dr1PollDelay = 1.0 / CLIENT_KNOBS - > BACKUP_AGGREGATE_POLL_RATE ;
state double dr2PollDelay = 1.0 / CLIENT_KNOBS - > BACKUP_AGGREGATE_POLL_RATE ;
agentFutures . push_back ( extraAgent . run ( cx , & dr1PollDelay , CLIENT_KNOBS - > SIM_BACKUP_TASKS_PER_AGENT ) ) ;
agentFutures . push_back ( dbAgent . run ( extraDB , & dr2PollDelay , CLIENT_KNOBS - > SIM_BACKUP_TASKS_PER_AGENT ) ) ;
2020-11-23 11:49:16 +08:00
while ( g_simulator . drAgents = = ISimulator : : BackupAgentType : : BackupToDB ) {
2018-08-11 04:57:10 +08:00
wait ( delay ( 1.0 ) ) ;
2017-05-26 04:48:44 +08:00
}
2018-02-21 05:22:31 +08:00
TraceEvent ( " StoppingDrAgents " ) ;
2017-05-26 04:48:44 +08:00
for ( auto it : agentFutures ) {
it . cancel ( ) ;
}
}
2018-08-11 08:25:43 +08:00
wait ( Future < Void > ( Never ( ) ) ) ;
2017-05-26 04:48:44 +08:00
throw internal_error ( ) ;
}
2019-07-24 10:24:07 +08:00
enum AgentMode {
AgentNone = 0 ,
AgentOnly = 1 ,
AgentAddition = 2
} ;
2017-05-26 04:48:44 +08:00
// SOMEDAY: when a process can be rebooted in isolation from the other on that machine,
// a loop{} will be needed around the waiting on simulatedFDBD(). For now this simply
// takes care of house-keeping such as context switching and file closing.
2019-02-28 16:09:53 +08:00
ACTOR Future < ISimulator : : KillType > simulatedFDBDRebooter ( Reference < ClusterConnectionFile > connFile , IPAddress ip ,
2020-02-07 11:29:31 +08:00
bool sslEnabled ,
2019-02-28 16:09:53 +08:00
uint16_t port , uint16_t listenPerProcess ,
LocalityData localities , ProcessClass processClass ,
std : : string * dataFolder , std : : string * coordFolder ,
std : : string baseFolder , ClusterConnectionString connStr ,
2019-07-24 10:24:07 +08:00
bool useSeedFile , AgentMode runBackupAgents ,
2019-04-13 04:23:02 +08:00
std : : string whitelistBinPaths ) {
2017-05-26 04:48:44 +08:00
state ISimulator : : ProcessInfo * simProcess = g_simulator . getCurrentProcess ( ) ;
2019-05-11 05:01:52 +08:00
state UID randomId = nondeterministicRandom ( ) - > randomUniqueID ( ) ;
2017-06-20 07:48:15 +08:00
state int cycles = 0 ;
2017-05-26 04:48:44 +08:00
loop {
2019-05-11 05:01:52 +08:00
auto waitTime = SERVER_KNOBS - > MIN_REBOOT_TIME + ( SERVER_KNOBS - > MAX_REBOOT_TIME - SERVER_KNOBS - > MIN_REBOOT_TIME ) * deterministicRandom ( ) - > random01 ( ) ;
2017-05-26 04:48:44 +08:00
cycles + + ;
2017-08-29 02:25:37 +08:00
TraceEvent ( " SimulatedFDBDPreWait " ) . detail ( " Cycles " , cycles ) . detail ( " RandomId " , randomId )
. detail ( " Address " , NetworkAddress ( ip , port , true , false ) )
2019-03-15 09:40:28 +08:00
. detail ( " ZoneId " , localities . zoneId ( ) )
2018-06-09 02:11:08 +08:00
. detail ( " WaitTime " , waitTime ) . detail ( " Port " , port ) ;
2017-05-26 04:48:44 +08:00
2018-08-11 04:57:10 +08:00
wait ( delay ( waitTime ) ) ;
2017-05-26 04:48:44 +08:00
2019-01-30 00:43:57 +08:00
state ISimulator : : ProcessInfo * process =
2020-02-11 06:00:21 +08:00
g_simulator . newProcess ( " Server " , ip , port , sslEnabled , listenPerProcess , localities , processClass , dataFolder - > c_str ( ) ,
2019-04-15 02:11:08 +08:00
coordFolder - > c_str ( ) ) ;
2019-01-30 00:43:57 +08:00
wait ( g_simulator . onProcess ( process ,
2019-06-25 17:47:35 +08:00
TaskPriority : : DefaultYield ) ) ; // Now switch execution to the process on which we will run
2017-05-26 04:48:44 +08:00
state Future < ISimulator : : KillType > onShutdown = process - > onShutdown ( ) ;
try {
2018-08-02 05:30:57 +08:00
TraceEvent ( " SimulatedRebooterStarting " ) . detail ( " Cycles " , cycles ) . detail ( " RandomId " , randomId )
2019-03-15 09:40:28 +08:00
. detail ( " ZoneId " , localities . zoneId ( ) )
. detail ( " DataHall " , localities . dataHallId ( ) )
2017-08-29 02:25:37 +08:00
. detail ( " Address " , process - > address . toString ( ) )
. detail ( " Excluded " , process - > excluded )
2018-04-26 09:29:29 +08:00
. detail ( " UsingSSL " , sslEnabled ) ;
2017-08-29 02:25:37 +08:00
TraceEvent ( " ProgramStart " ) . detail ( " Cycles " , cycles ) . detail ( " RandomId " , randomId )
2019-11-16 04:26:51 +08:00
. detail ( " SourceVersion " , getSourceVersion ( ) )
2017-05-26 04:48:44 +08:00
. detail ( " Version " , FDB_VT_VERSION )
. detail ( " PackageName " , FDB_VT_PACKAGE_NAME )
. detail ( " DataFolder " , * dataFolder )
. detail ( " ConnectionString " , connFile ? connFile - > getConnectionString ( ) . toString ( ) : " " )
2020-08-19 05:18:50 +08:00
. detailf ( " ActualTime " , " %lld " , DEBUG_DETERMINISM ? 0 : time ( nullptr ) )
2017-05-26 04:48:44 +08:00
. detail ( " CommandLine " , " fdbserver -r simulation " )
2019-04-04 08:37:14 +08:00
. detail ( " BuggifyEnabled " , isBuggifyEnabled ( BuggifyType : : General ) )
2017-05-26 04:48:44 +08:00
. detail ( " Simulated " , true )
. trackLatest ( " ProgramStart " ) ;
try {
//SOMEDAY: test lower memory limits, without making them too small and causing the database to stop making progress
2019-07-24 10:24:07 +08:00
FlowTransport : : createInstance ( processClass = = ProcessClass : : TesterClass | | runBackupAgents = = AgentOnly , 1 ) ;
2017-05-26 04:48:44 +08:00
Sim2FileSystem : : newFileSystem ( ) ;
2018-12-07 03:48:50 +08:00
vector < Future < Void > > futures ;
for ( int listenPort = port ; listenPort < port + listenPerProcess ; + + listenPort ) {
2019-02-01 10:20:14 +08:00
NetworkAddress n ( ip , listenPort , true , sslEnabled & & listenPort = = port ) ;
2018-12-07 03:48:50 +08:00
futures . push_back ( FlowTransport : : transport ( ) . bind ( n , n ) ) ;
}
2019-07-24 10:24:07 +08:00
if ( runBackupAgents ! = AgentOnly ) {
futures . push_back ( fdbd ( connFile , localities , processClass , * dataFolder , * coordFolder , 500e6 , " " , " " , - 1 , whitelistBinPaths ) ) ;
}
if ( runBackupAgents ! = AgentNone ) {
futures . push_back ( runBackup ( connFile ) ) ;
futures . push_back ( runDr ( connFile ) ) ;
}
2017-05-26 04:48:44 +08:00
2018-12-07 03:48:50 +08:00
futures . push_back ( success ( onShutdown ) ) ;
wait ( waitForAny ( futures ) ) ;
2017-05-26 04:48:44 +08:00
} catch ( Error & e ) {
2017-05-27 08:43:28 +08:00
// If in simulation, if we make it here with an error other than io_timeout but enASIOTimedOut is set then somewhere an io_timeout was converted to a different error.
if ( g_network - > isSimulated ( ) & & e . code ( ) ! = error_code_io_timeout & & ( bool ) g_network - > global ( INetwork : : enASIOTimedOut ) )
2017-08-29 02:25:37 +08:00
TraceEvent ( SevError , " IOTimeoutErrorSuppressed " ) . detail ( " ErrorCode " , e . code ( ) ) . detail ( " RandomId " , randomId ) . backtrace ( ) ;
2017-05-27 08:43:28 +08:00
2017-05-26 04:48:44 +08:00
if ( onShutdown . isReady ( ) & & onShutdown . isError ( ) ) throw onShutdown . getError ( ) ;
if ( e . code ( ) ! = error_code_actor_cancelled )
printf ( " SimulatedFDBDTerminated: %s \n " , e . what ( ) ) ;
ASSERT ( destructed | | g_simulator . getCurrentProcess ( ) = = process ) ; // simulatedFDBD catch called on different process
2019-03-15 09:40:28 +08:00
TraceEvent ( e . code ( ) = = error_code_actor_cancelled | | e . code ( ) = = error_code_file_not_found | | destructed ? SevInfo : SevError , " SimulatedFDBDTerminated " ) . error ( e , true ) . detail ( " ZoneId " , localities . zoneId ( ) ) ;
2017-05-26 04:48:44 +08:00
}
2018-08-02 05:30:57 +08:00
TraceEvent ( " SimulatedFDBDDone " ) . detail ( " Cycles " , cycles ) . detail ( " RandomId " , randomId )
2017-08-29 02:25:37 +08:00
. detail ( " Address " , process - > address )
. detail ( " Excluded " , process - > excluded )
2019-03-15 09:40:28 +08:00
. detail ( " ZoneId " , localities . zoneId ( ) )
2017-05-26 04:48:44 +08:00
. detail ( " KillType " , onShutdown . isReady ( ) ? onShutdown . get ( ) : ISimulator : : None ) ;
if ( ! onShutdown . isReady ( ) )
onShutdown = ISimulator : : InjectFaults ;
} catch ( Error & e ) {
2019-03-15 09:40:28 +08:00
TraceEvent ( destructed ? SevInfo : SevError , " SimulatedFDBDRebooterError " ) . error ( e , true ) . detail ( " ZoneId " , localities . zoneId ( ) ) . detail ( " RandomId " , randomId ) ;
2017-05-26 04:48:44 +08:00
onShutdown = e ;
}
ASSERT ( destructed | | g_simulator . getCurrentProcess ( ) = = process ) ;
if ( ! process - > shutdownSignal . isSet ( ) & & ! destructed ) {
process - > rebooting = true ;
process - > shutdownSignal . send ( ISimulator : : None ) ;
}
2018-08-02 05:30:57 +08:00
TraceEvent ( " SimulatedFDBDWait " ) . detail ( " Cycles " , cycles ) . detail ( " RandomId " , randomId )
2017-08-29 02:25:37 +08:00
. detail ( " Address " , process - > address )
. detail ( " Excluded " , process - > excluded )
. detail ( " Rebooting " , process - > rebooting )
2019-03-15 09:40:28 +08:00
. detail ( " ZoneId " , localities . zoneId ( ) ) ;
2018-08-11 04:57:10 +08:00
wait ( g_simulator . onProcess ( simProcess ) ) ;
2017-05-26 04:48:44 +08:00
2018-08-11 04:57:10 +08:00
wait ( delay ( 0.00001 + FLOW_KNOBS - > MAX_BUGGIFIED_DELAY ) ) ; // One last chance for the process to clean up?
2017-05-26 04:48:44 +08:00
g_simulator . destroyProcess ( process ) ; // Leak memory here; the process may be used in other parts of the simulation
auto shutdownResult = onShutdown . get ( ) ;
2018-08-02 05:30:57 +08:00
TraceEvent ( " SimulatedFDBDShutdown " ) . detail ( " Cycles " , cycles ) . detail ( " RandomId " , randomId )
2017-08-29 02:25:37 +08:00
. detail ( " Address " , process - > address )
. detail ( " Excluded " , process - > excluded )
2019-03-15 09:40:28 +08:00
. detail ( " ZoneId " , localities . zoneId ( ) )
2017-05-26 04:48:44 +08:00
. detail ( " KillType " , shutdownResult ) ;
if ( shutdownResult < ISimulator : : RebootProcessAndDelete ) {
2018-08-02 05:30:57 +08:00
TraceEvent ( " SimulatedFDBDLowerReboot " ) . detail ( " Cycles " , cycles ) . detail ( " RandomId " , randomId )
2017-08-29 02:25:37 +08:00
. detail ( " Address " , process - > address )
. detail ( " Excluded " , process - > excluded )
2019-03-15 09:40:28 +08:00
. detail ( " ZoneId " , localities . zoneId ( ) )
2017-05-26 04:48:44 +08:00
. detail ( " KillType " , shutdownResult ) ;
return onShutdown . get ( ) ;
}
if ( onShutdown . get ( ) = = ISimulator : : RebootProcessAndDelete ) {
2018-08-02 05:30:57 +08:00
TraceEvent ( " SimulatedFDBDRebootAndDelete " ) . detail ( " Cycles " , cycles ) . detail ( " RandomId " , randomId )
2017-08-29 02:25:37 +08:00
. detail ( " Address " , process - > address )
2019-03-15 09:40:28 +08:00
. detail ( " ZoneId " , localities . zoneId ( ) )
2017-05-26 04:48:44 +08:00
. detail ( " KillType " , shutdownResult ) ;
2019-05-11 05:01:52 +08:00
* coordFolder = joinPath ( baseFolder , deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ) ;
* dataFolder = joinPath ( baseFolder , deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ) ;
2017-05-26 04:48:44 +08:00
platform : : createDirectory ( * dataFolder ) ;
if ( ! useSeedFile ) {
writeFile ( joinPath ( * dataFolder , " fdb.cluster " ) , connStr . toString ( ) ) ;
2020-11-07 15:50:55 +08:00
connFile = makeReference < ClusterConnectionFile > ( joinPath ( * dataFolder , " fdb.cluster " ) ) ;
2017-05-26 04:48:44 +08:00
}
else {
2020-11-07 15:50:55 +08:00
connFile =
makeReference < ClusterConnectionFile > ( joinPath ( * dataFolder , " fdb.cluster " ) , connStr . toString ( ) ) ;
2017-05-26 04:48:44 +08:00
}
}
else {
2018-08-02 05:30:57 +08:00
TraceEvent ( " SimulatedFDBDJustRepeat " ) . detail ( " Cycles " , cycles ) . detail ( " RandomId " , randomId )
2017-08-29 02:25:37 +08:00
. detail ( " Address " , process - > address )
2019-03-15 09:40:28 +08:00
. detail ( " ZoneId " , localities . zoneId ( ) )
2017-05-26 04:48:44 +08:00
. detail ( " KillType " , shutdownResult ) ;
}
}
}
template < >
std : : string describe ( bool const & val ) {
return val ? " true " : " false " ;
}
template < >
std : : string describe ( int const & val ) {
return format ( " %d " , val ) ;
}
// Since a datacenter kill is considered to be the same as killing a machine, files cannot be swapped across datacenters
std : : map < Optional < Standalone < StringRef > > , std : : vector < std : : vector < std : : string > > > availableFolders ;
// process count is no longer needed because it is now the length of the vector of ip's, because it was one ip per process
2020-02-07 11:29:31 +08:00
ACTOR Future < Void > simulatedMachine ( ClusterConnectionString connStr , std : : vector < IPAddress > ips , bool sslEnabled , LocalityData localities ,
2019-02-28 16:09:53 +08:00
ProcessClass processClass , std : : string baseFolder , bool restarting ,
2019-07-24 10:24:07 +08:00
bool useSeedFile , AgentMode runBackupAgents , bool sslOnly , std : : string whitelistBinPaths ) {
2017-05-26 04:48:44 +08:00
state int bootCount = 0 ;
state std : : vector < std : : string > myFolders ;
state std : : vector < std : : string > coordFolders ;
2019-05-11 05:01:52 +08:00
state UID randomId = nondeterministicRandom ( ) - > randomUniqueID ( ) ;
2019-02-01 10:20:14 +08:00
state int listenPerProcess = ( sslEnabled & & ! sslOnly ) ? 2 : 1 ;
2017-05-26 04:48:44 +08:00
try {
CSimpleIni ini ;
ini . SetUnicode ( ) ;
ini . LoadFile ( joinPath ( baseFolder , " restartInfo.ini " ) . c_str ( ) ) ;
for ( int i = 0 ; i < ips . size ( ) ; i + + ) {
if ( restarting ) {
2019-05-11 05:01:52 +08:00
myFolders . push_back ( ini . GetValue ( printable ( localities . machineId ( ) ) . c_str ( ) , format ( " %d " , i * listenPerProcess ) . c_str ( ) , joinPath ( baseFolder , deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ) . c_str ( ) ) ) ;
2017-05-26 04:48:44 +08:00
if ( i = = 0 ) {
2019-01-19 07:42:48 +08:00
std : : string coordinationFolder = ini . GetValue ( printable ( localities . machineId ( ) ) . c_str ( ) , " coordinationFolder " , " " ) ;
2017-05-26 04:48:44 +08:00
if ( ! coordinationFolder . size ( ) )
2019-05-11 05:01:52 +08:00
coordinationFolder = ini . GetValue ( printable ( localities . machineId ( ) ) . c_str ( ) , format ( " c%d " , i * listenPerProcess ) . c_str ( ) , joinPath ( baseFolder , deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ) . c_str ( ) ) ;
2017-05-26 04:48:44 +08:00
coordFolders . push_back ( coordinationFolder ) ;
} else {
2019-05-11 05:01:52 +08:00
coordFolders . push_back ( ini . GetValue ( printable ( localities . machineId ( ) ) . c_str ( ) , format ( " c%d " , i * listenPerProcess ) . c_str ( ) , joinPath ( baseFolder , deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ) . c_str ( ) ) ) ;
2017-05-26 04:48:44 +08:00
}
}
else {
2019-05-11 05:01:52 +08:00
coordFolders . push_back ( joinPath ( baseFolder , deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ) ) ;
std : : string thisFolder = deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ;
2017-05-26 04:48:44 +08:00
myFolders . push_back ( joinPath ( baseFolder , thisFolder ) ) ;
platform : : createDirectory ( myFolders [ i ] ) ;
if ( ! useSeedFile )
writeFile ( joinPath ( myFolders [ i ] , " fdb.cluster " ) , connStr . toString ( ) ) ;
}
}
loop {
state std : : vector < Future < ISimulator : : KillType > > processes ;
for ( int i = 0 ; i < ips . size ( ) ; i + + ) {
std : : string path = joinPath ( myFolders [ i ] , " fdb.cluster " ) ;
Reference < ClusterConnectionFile > clusterFile ( useSeedFile ? new ClusterConnectionFile ( path , connStr . toString ( ) ) : new ClusterConnectionFile ( path ) ) ;
2019-01-09 23:41:02 +08:00
const int listenPort = i * listenPerProcess + 1 ;
2019-07-24 10:24:07 +08:00
AgentMode agentMode = runBackupAgents = = AgentOnly ? ( i = = ips . size ( ) - 1 ? AgentOnly : AgentNone ) : runBackupAgents ;
2020-02-07 11:29:31 +08:00
processes . push_back ( simulatedFDBDRebooter ( clusterFile , ips [ i ] , sslEnabled , listenPort , listenPerProcess , localities , processClass , & myFolders [ i ] , & coordFolders [ i ] , baseFolder , connStr , useSeedFile , agentMode , whitelistBinPaths ) ) ;
2019-03-15 09:40:28 +08:00
TraceEvent ( " SimulatedMachineProcess " , randomId ) . detail ( " Address " , NetworkAddress ( ips [ i ] , listenPort , true , false ) ) . detail ( " ZoneId " , localities . zoneId ( ) ) . detail ( " DataHall " , localities . dataHallId ( ) ) . detail ( " Folder " , myFolders [ i ] ) ;
2017-05-26 04:48:44 +08:00
}
TEST ( bootCount > = 1 ) ; // Simulated machine rebooted
TEST ( bootCount > = 2 ) ; // Simulated machine rebooted twice
TEST ( bootCount > = 3 ) ; // Simulated machine rebooted three times
+ + bootCount ;
2017-08-29 02:25:37 +08:00
TraceEvent ( " SimulatedMachineStart " , randomId )
2017-05-26 04:48:44 +08:00
. detail ( " Folder0 " , myFolders [ 0 ] )
. detail ( " CFolder0 " , coordFolders [ 0 ] )
. detail ( " MachineIPs " , toIPVectorString ( ips ) )
. detail ( " SSL " , sslEnabled )
2018-06-09 02:11:08 +08:00
. detail ( " Processes " , processes . size ( ) )
. detail ( " BootCount " , bootCount )
2017-05-26 04:48:44 +08:00
. detail ( " ProcessClass " , processClass . toString ( ) )
. detail ( " Restarting " , restarting )
. detail ( " UseSeedFile " , useSeedFile )
2019-03-15 09:40:28 +08:00
. detail ( " ZoneId " , localities . zoneId ( ) )
. detail ( " DataHall " , localities . dataHallId ( ) )
2017-05-26 04:48:44 +08:00
. detail ( " Locality " , localities . toString ( ) ) ;
2018-08-11 04:57:10 +08:00
wait ( waitForAll ( processes ) ) ;
2017-05-26 04:48:44 +08:00
2017-08-29 02:25:37 +08:00
TraceEvent ( " SimulatedMachineRebootStart " , randomId )
2017-05-26 04:48:44 +08:00
. detail ( " Folder0 " , myFolders [ 0 ] )
. detail ( " CFolder0 " , coordFolders [ 0 ] )
. detail ( " MachineIPs " , toIPVectorString ( ips ) )
2019-03-15 09:40:28 +08:00
. detail ( " ZoneId " , localities . zoneId ( ) )
. detail ( " DataHall " , localities . dataHallId ( ) ) ;
2017-05-26 04:48:44 +08:00
2019-02-18 10:46:59 +08:00
{
//Kill all open files, which may cause them to write invalid data.
auto & machineCache = g_simulator . getMachineById ( localities . machineId ( ) ) - > openFiles ;
2017-05-26 04:48:44 +08:00
2019-02-18 10:46:59 +08:00
//Copy the file pointers to a vector because the map may be modified while we are killing files
std : : vector < AsyncFileNonDurable * > files ;
for ( auto fileItr = machineCache . begin ( ) ; fileItr ! = machineCache . end ( ) ; + + fileItr ) {
ASSERT ( fileItr - > second . isReady ( ) ) ;
files . push_back ( ( AsyncFileNonDurable * ) fileItr - > second . get ( ) . getPtr ( ) ) ;
}
2017-05-26 04:48:44 +08:00
2019-02-18 10:46:59 +08:00
std : : vector < Future < Void > > killFutures ;
for ( auto fileItr = files . begin ( ) ; fileItr ! = files . end ( ) ; + + fileItr )
killFutures . push_back ( ( * fileItr ) - > kill ( ) ) ;
2017-05-26 04:48:44 +08:00
2019-02-18 10:46:59 +08:00
wait ( waitForAll ( killFutures ) ) ;
}
2017-05-26 04:48:44 +08:00
state std : : set < std : : string > filenames ;
state std : : string closingStr ;
2019-01-19 07:42:48 +08:00
auto & machineCache = g_simulator . getMachineById ( localities . machineId ( ) ) - > openFiles ;
2017-05-26 04:48:44 +08:00
for ( auto it : machineCache ) {
filenames . insert ( it . first ) ;
closingStr + = it . first + " , " ;
ASSERT ( it . second . isReady ( ) & & ! it . second . isError ( ) ) ;
}
2019-01-19 07:42:48 +08:00
for ( auto it : g_simulator . getMachineById ( localities . machineId ( ) ) - > deletingFiles ) {
2017-08-26 01:12:58 +08:00
filenames . insert ( it ) ;
closingStr + = it + " , " ;
}
2017-08-29 02:25:37 +08:00
TraceEvent ( " SimulatedMachineRebootAfterKills " , randomId )
2017-05-26 04:48:44 +08:00
. detail ( " Folder0 " , myFolders [ 0 ] )
. detail ( " CFolder0 " , coordFolders [ 0 ] )
. detail ( " MachineIPs " , toIPVectorString ( ips ) )
. detail ( " Closing " , closingStr )
2019-03-15 09:40:28 +08:00
. detail ( " ZoneId " , localities . zoneId ( ) )
. detail ( " DataHall " , localities . dataHallId ( ) ) ;
2017-05-26 04:48:44 +08:00
2019-01-19 07:42:48 +08:00
ISimulator : : MachineInfo * machine = g_simulator . getMachineById ( localities . machineId ( ) ) ;
2017-05-26 04:48:44 +08:00
machine - > closingFiles = filenames ;
2019-01-19 07:42:48 +08:00
g_simulator . getMachineById ( localities . machineId ( ) ) - > openFiles . clear ( ) ;
2017-05-26 04:48:44 +08:00
// During a reboot:
// The process is expected to close all files and be inactive in zero time, but not necessarily
// without delay(0)-equivalents, so delay(0) a few times waiting for it to achieve that goal.
// After an injected fault:
// The process is expected to shut down eventually, but not necessarily instantly. Wait up to 60 seconds.
state int shutdownDelayCount = 0 ;
state double backoff = 0 ;
loop {
2019-01-19 07:42:48 +08:00
auto & machineCache = g_simulator . getMachineById ( localities . machineId ( ) ) - > closingFiles ;
2017-05-26 04:48:44 +08:00
if ( ! machineCache . empty ( ) ) {
std : : string openFiles ;
int i = 0 ;
for ( auto it = machineCache . begin ( ) ; it ! = machineCache . end ( ) & & i < 5 ; + + it ) {
openFiles + = * it + " , " ;
i + + ;
}
2017-08-29 02:25:37 +08:00
TraceEvent ( " MachineFilesOpen " , randomId ) . detail ( " PAddr " , toIPVectorString ( ips ) ) . detail ( " OpenFiles " , openFiles ) ;
2017-05-26 04:48:44 +08:00
} else
break ;
if ( shutdownDelayCount + + > = 50 ) { // Worker doesn't shut down instantly on reboot
2017-08-29 02:25:37 +08:00
TraceEvent ( SevError , " SimulatedFDBDFilesCheck " , randomId )
2017-05-26 04:48:44 +08:00
. detail ( " PAddrs " , toIPVectorString ( ips ) )
2019-03-15 09:40:28 +08:00
. detail ( " ZoneId " , localities . zoneId ( ) )
. detail ( " DataHall " , localities . dataHallId ( ) ) ;
2017-05-26 04:48:44 +08:00
ASSERT ( false ) ;
}
2018-08-11 04:57:10 +08:00
wait ( delay ( backoff ) ) ;
2017-05-26 04:48:44 +08:00
backoff = std : : min ( backoff + 1.0 , 6.0 ) ;
}
2017-08-29 02:25:37 +08:00
TraceEvent ( " SimulatedFDBDFilesClosed " , randomId )
. detail ( " Address " , toIPVectorString ( ips ) )
2019-03-15 09:40:28 +08:00
. detail ( " ZoneId " , localities . zoneId ( ) )
. detail ( " DataHall " , localities . dataHallId ( ) ) ;
2017-05-26 04:48:44 +08:00
2019-01-19 07:42:48 +08:00
g_simulator . destroyMachine ( localities . machineId ( ) ) ;
2017-05-26 04:48:44 +08:00
// SOMEDAY: when processes can be rebooted, this check will be needed
//ASSERT( this machine is rebooting );
// Since processes can end with different codes, take the highest (least severe) to detmine what to do
state ISimulator : : KillType killType = processes [ 0 ] . get ( ) ;
for ( int i = 1 ; i < ips . size ( ) ; i + + )
killType = std : : max ( processes [ i ] . get ( ) , killType ) ;
TEST ( true ) ; // Simulated machine has been rebooted
state bool swap = killType = = ISimulator : : Reboot & & BUGGIFY_WITH_PROB ( 0.75 ) & & g_simulator . canSwapToMachine ( localities . zoneId ( ) ) ;
if ( swap )
availableFolders [ localities . dcId ( ) ] . push_back ( myFolders ) ;
2019-05-11 05:01:52 +08:00
auto rebootTime = deterministicRandom ( ) - > random01 ( ) * MACHINE_REBOOT_TIME ;
2017-05-26 04:48:44 +08:00
2017-08-29 02:25:37 +08:00
TraceEvent ( " SimulatedMachineShutdown " , randomId )
2017-05-26 04:48:44 +08:00
. detail ( " Swap " , swap )
. detail ( " KillType " , killType )
. detail ( " RebootTime " , rebootTime )
2019-03-15 09:40:28 +08:00
. detail ( " ZoneId " , localities . zoneId ( ) )
. detail ( " DataHall " , localities . dataHallId ( ) )
2017-05-26 04:48:44 +08:00
. detail ( " MachineIPs " , toIPVectorString ( ips ) ) ;
2018-08-11 04:57:10 +08:00
wait ( delay ( rebootTime ) ) ;
2017-05-26 04:48:44 +08:00
if ( swap ) {
auto & avail = availableFolders [ localities . dcId ( ) ] ;
2019-05-11 05:01:52 +08:00
int i = deterministicRandom ( ) - > randomInt ( 0 , avail . size ( ) ) ;
2017-05-26 04:48:44 +08:00
if ( i ! = avail . size ( ) - 1 )
std : : swap ( avail [ i ] , avail . back ( ) ) ;
auto toRebootFrom = avail . back ( ) ;
avail . pop_back ( ) ;
if ( myFolders ! = toRebootFrom ) {
TEST ( true ) ; // Simulated machine swapped data folders
2017-08-29 02:25:37 +08:00
TraceEvent ( " SimulatedMachineFolderSwap " , randomId )
2017-05-26 04:48:44 +08:00
. detail ( " OldFolder0 " , myFolders [ 0 ] ) . detail ( " NewFolder0 " , toRebootFrom [ 0 ] )
. detail ( " MachineIPs " , toIPVectorString ( ips ) ) ;
}
myFolders = toRebootFrom ;
if ( ! useSeedFile ) {
for ( auto f : toRebootFrom ) {
if ( ! fileExists ( joinPath ( f , " fdb.cluster " ) ) ) {
writeFile ( joinPath ( f , " fdb.cluster " ) , connStr . toString ( ) ) ;
}
}
}
} else if ( killType = = ISimulator : : RebootAndDelete ) {
for ( int i = 0 ; i < ips . size ( ) ; i + + ) {
2019-05-11 05:01:52 +08:00
coordFolders [ i ] = joinPath ( baseFolder , deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ) ;
myFolders [ i ] = joinPath ( baseFolder , deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ) ;
2017-05-26 04:48:44 +08:00
platform : : createDirectory ( myFolders [ i ] ) ;
if ( ! useSeedFile ) {
writeFile ( joinPath ( myFolders [ i ] , " fdb.cluster " ) , connStr . toString ( ) ) ;
}
}
TEST ( true ) ; // Simulated machine rebooted with data loss
}
//this machine is rebooting = false;
}
} catch ( Error & e ) {
2019-01-19 07:42:48 +08:00
g_simulator . getMachineById ( localities . machineId ( ) ) - > openFiles . clear ( ) ;
2017-05-26 04:48:44 +08:00
throw ;
}
}
2019-02-28 16:09:53 +08:00
IPAddress makeIPAddressForSim ( bool isIPv6 , std : : array < int , 4 > parts ) {
if ( isIPv6 ) {
IPAddress : : IPAddressStore addrStore { 0xAB , 0xCD } ;
uint16_t * ptr = ( uint16_t * ) addrStore . data ( ) ;
ptr [ 4 ] = ( uint16_t ) ( parts [ 0 ] < < 8 ) ;
ptr [ 5 ] = ( uint16_t ) ( parts [ 1 ] < < 8 ) ;
ptr [ 6 ] = ( uint16_t ) ( parts [ 2 ] < < 8 ) ;
ptr [ 7 ] = ( uint16_t ) ( parts [ 3 ] < < 8 ) ;
return IPAddress ( addrStore ) ;
} else {
return IPAddress ( parts [ 0 ] < < 24 | parts [ 1 ] < < 16 | parts [ 2 ] < < 8 | parts [ 3 ] ) ;
}
}
2017-05-26 04:48:44 +08:00
# include "fdbclient/MonitorLeader.h"
2019-01-30 00:43:57 +08:00
ACTOR Future < Void > restartSimulatedSystem ( vector < Future < Void > > * systemActors , std : : string baseFolder , int * pTesterCount ,
Optional < ClusterConnectionString > * pConnString ,
Standalone < StringRef > * pStartingConfiguration ,
2020-02-07 11:29:31 +08:00
int extraDB , std : : string whitelistBinPaths ) {
2017-05-26 04:48:44 +08:00
CSimpleIni ini ;
ini . SetUnicode ( ) ;
ini . LoadFile ( joinPath ( baseFolder , " restartInfo.ini " ) . c_str ( ) ) ;
// allows multiple ipAddr entries
ini . SetMultiKey ( ) ;
try {
int machineCount = atoi ( ini . GetValue ( " META " , " machineCount " ) ) ;
int processesPerMachine = atoi ( ini . GetValue ( " META " , " processesPerMachine " ) ) ;
2019-02-01 10:20:14 +08:00
int listenersPerProcess = 1 ;
auto listenersPerProcessStr = ini . GetValue ( " META " , " listenersPerProcess " ) ;
2020-08-19 05:18:50 +08:00
if ( listenersPerProcessStr ! = nullptr ) {
2019-02-01 10:20:14 +08:00
listenersPerProcess = atoi ( listenersPerProcessStr ) ;
}
2017-05-26 04:48:44 +08:00
int desiredCoordinators = atoi ( ini . GetValue ( " META " , " desiredCoordinators " ) ) ;
int testerCount = atoi ( ini . GetValue ( " META " , " testerCount " ) ) ;
2018-04-28 04:54:34 +08:00
bool enableExtraDB = ( extraDB = = 3 ) ;
2017-05-26 04:48:44 +08:00
ClusterConnectionString conn ( ini . GetValue ( " META " , " connectionString " ) ) ;
2018-04-28 04:54:34 +08:00
if ( enableExtraDB ) {
g_simulator . extraDB = new ClusterConnectionString ( ini . GetValue ( " META " , " connectionString " ) ) ;
}
2017-05-26 04:48:44 +08:00
* pConnString = conn ;
* pTesterCount = testerCount ;
2019-02-01 10:20:14 +08:00
bool usingSSL = conn . toString ( ) . find ( " :tls " ) ! = std : : string : : npos | | listenersPerProcess > 1 ;
2019-05-11 05:01:52 +08:00
int useSeedForMachine = deterministicRandom ( ) - > randomInt ( 0 , machineCount ) ;
2018-11-03 05:16:47 +08:00
std : : vector < std : : string > dcIds ;
2017-05-26 04:48:44 +08:00
for ( int i = 0 ; i < machineCount ; i + + ) {
Optional < Standalone < StringRef > > dcUID ;
2019-01-19 07:42:48 +08:00
Optional < Standalone < StringRef > > zoneId ;
std : : string machineIdString = ini . GetValue ( " META " , format ( " %d " , i ) . c_str ( ) ) ;
Standalone < StringRef > machineId = StringRef ( machineIdString ) ;
std : : string dcUIDini = ini . GetValue ( machineIdString . c_str ( ) , " dcUID " ) ;
2018-11-03 05:16:47 +08:00
if ( ! dcUIDini . empty ( ) ) {
dcUID = StringRef ( dcUIDini ) ;
}
2019-01-19 07:42:48 +08:00
auto zoneIDini = ini . GetValue ( machineIdString . c_str ( ) , " zoneId " ) ;
2020-08-19 05:18:50 +08:00
if ( zoneIDini = = nullptr ) {
2019-01-19 07:42:48 +08:00
zoneId = machineId ;
} else {
zoneId = StringRef ( zoneIDini ) ;
}
ProcessClass processClass = ProcessClass ( ( ProcessClass : : ClassType ) atoi ( ini . GetValue ( machineIdString . c_str ( ) , " mClass " ) ) , ProcessClass : : CommandLineSource ) ;
2018-11-03 05:16:47 +08:00
if ( processClass ! = ProcessClass : : TesterClass ) {
dcIds . push_back ( dcUIDini ) ;
}
2017-05-26 04:48:44 +08:00
2019-02-28 16:09:53 +08:00
std : : vector < IPAddress > ipAddrs ;
2019-01-19 07:42:48 +08:00
int processes = atoi ( ini . GetValue ( machineIdString . c_str ( ) , " processes " ) ) ;
2017-05-26 04:48:44 +08:00
2019-01-19 07:42:48 +08:00
auto ip = ini . GetValue ( machineIdString . c_str ( ) , " ipAddr " ) ;
2017-05-26 04:48:44 +08:00
2019-03-01 04:24:56 +08:00
// Helper to translate the IP address stored in INI file to out IPAddress representation.
// After IPv6 work, we store the actual string representation of IP address, however earlier, it was
// instead the 32 bit integer value.
2019-02-28 16:09:53 +08:00
auto parseIp = [ ] ( const char * ipStr ) - > IPAddress {
Optional < IPAddress > parsedIp = IPAddress : : parse ( ipStr ) ;
if ( parsedIp . present ( ) ) {
return parsedIp . get ( ) ;
} else {
2020-08-19 05:18:50 +08:00
return IPAddress ( strtoul ( ipStr , nullptr , 10 ) ) ;
2019-02-28 16:09:53 +08:00
}
} ;
2020-08-19 05:18:50 +08:00
if ( ip = = nullptr ) {
2019-02-28 16:09:53 +08:00
for ( int i = 0 ; i < processes ; i + + ) {
const char * val =
ini . GetValue ( machineIdString . c_str ( ) , format ( " ipAddr%d " , i * listenersPerProcess ) . c_str ( ) ) ;
ipAddrs . push_back ( parseIp ( val ) ) ;
2017-05-26 04:48:44 +08:00
}
}
else {
// old way
2019-02-28 16:09:53 +08:00
ipAddrs . push_back ( parseIp ( ip ) ) ;
2017-05-26 04:48:44 +08:00
for ( int i = 1 ; i < processes ; i + + ) {
2019-02-28 16:09:53 +08:00
if ( ipAddrs . back ( ) . isV6 ( ) ) {
IPAddress : : IPAddressStore store = ipAddrs . back ( ) . toV6 ( ) ;
uint16_t * ptr = ( uint16_t * ) store . data ( ) ;
ptr [ 7 ] + = 1 ;
ipAddrs . push_back ( IPAddress ( store ) ) ;
} else {
ipAddrs . push_back ( IPAddress ( ipAddrs . back ( ) . toV4 ( ) + 1 ) ) ;
}
2017-05-26 04:48:44 +08:00
}
}
2019-01-19 07:42:48 +08:00
LocalityData localities ( Optional < Standalone < StringRef > > ( ) , zoneId , machineId , dcUID ) ;
2017-05-26 04:48:44 +08:00
localities . set ( LiteralStringRef ( " data_hall " ) , dcUID ) ;
2018-04-28 04:54:34 +08:00
// SOMEDAY: parse backup agent from test file
2019-01-30 00:43:57 +08:00
systemActors - > push_back ( reportErrors (
2020-02-07 11:29:31 +08:00
simulatedMachine ( conn , ipAddrs , usingSSL , localities , processClass , baseFolder , true ,
2020-11-03 05:58:08 +08:00
i = = useSeedForMachine , AgentAddition ,
2020-10-01 02:24:52 +08:00
usingSSL & & ( listenersPerProcess = = 1 | | processClass = = ProcessClass : : TesterClass ) ,
whitelistBinPaths ) ,
2019-01-30 00:43:57 +08:00
processClass = = ProcessClass : : TesterClass ? " SimulatedTesterMachine " : " SimulatedMachine " ) ) ;
2017-05-26 04:48:44 +08:00
}
g_simulator . desiredCoordinators = desiredCoordinators ;
g_simulator . processesPerMachine = processesPerMachine ;
2018-11-03 05:16:47 +08:00
uniquify ( dcIds ) ;
if ( ! BUGGIFY & & dcIds . size ( ) = = 2 & & dcIds [ 0 ] ! = " " & & dcIds [ 1 ] ! = " " ) {
StatusObject primaryObj ;
StatusObject primaryDcObj ;
primaryDcObj [ " id " ] = dcIds [ 0 ] ;
primaryDcObj [ " priority " ] = 2 ;
StatusArray primaryDcArr ;
primaryDcArr . push_back ( primaryDcObj ) ;
StatusObject remoteObj ;
StatusObject remoteDcObj ;
remoteDcObj [ " id " ] = dcIds [ 1 ] ;
remoteDcObj [ " priority " ] = 1 ;
StatusArray remoteDcArr ;
remoteDcArr . push_back ( remoteDcObj ) ;
primaryObj [ " datacenters " ] = primaryDcArr ;
remoteObj [ " datacenters " ] = remoteDcArr ;
StatusArray regionArr ;
regionArr . push_back ( primaryObj ) ;
regionArr . push_back ( remoteObj ) ;
* pStartingConfiguration = " single usable_regions=2 regions= " + json_spirit : : write_string ( json_spirit : : mValue ( regionArr ) , json_spirit : : Output_options : : none ) ;
}
2019-02-05 03:39:06 +08:00
TraceEvent ( " RestartSimulatorSettings " )
. detail ( " DesiredCoordinators " , g_simulator . desiredCoordinators )
. detail ( " ProcessesPerMachine " , g_simulator . processesPerMachine )
. detail ( " ListenersPerProcess " , listenersPerProcess ) ;
2017-05-26 04:48:44 +08:00
}
catch ( Error & e ) {
2018-06-09 02:11:08 +08:00
TraceEvent ( SevError , " RestartSimulationError " ) . error ( e ) ;
2017-05-26 04:48:44 +08:00
}
2018-08-11 04:57:10 +08:00
wait ( delay ( 1.0 ) ) ;
2017-05-26 04:48:44 +08:00
return Void ( ) ;
}
2017-08-12 06:06:36 +08:00
struct SimulationConfig {
2018-09-18 09:32:39 +08:00
explicit SimulationConfig ( int extraDB , int minimumReplication , int minimumRegions ) ;
2017-08-22 13:29:56 +08:00
int extraDB ;
2017-08-12 06:06:36 +08:00
2017-08-22 13:29:56 +08:00
DatabaseConfiguration db ;
2017-08-12 06:06:36 +08:00
2017-08-22 13:29:56 +08:00
void set_config ( std : : string config ) ;
2017-08-12 06:06:36 +08:00
2017-08-22 13:29:56 +08:00
// Simulation layout
int datacenters ;
int machine_count ; // Total, not per DC.
int processes_per_machine ;
int coordinators ;
2017-08-12 06:06:36 +08:00
private :
2018-09-18 09:32:39 +08:00
void generateNormalConfig ( int minimumReplication , int minimumRegions ) ;
2017-08-12 06:06:36 +08:00
} ;
2018-09-18 09:32:39 +08:00
SimulationConfig : : SimulationConfig ( int extraDB , int minimumReplication , int minimumRegions ) : extraDB ( extraDB ) {
generateNormalConfig ( minimumReplication , minimumRegions ) ;
2017-08-22 13:29:56 +08:00
}
2017-08-12 06:06:36 +08:00
2017-08-22 13:29:56 +08:00
void SimulationConfig : : set_config ( std : : string config ) {
// The only mechanism we have for turning "single" into what single means
// is buildConfiguration()... :/
std : : map < std : : string , std : : string > hack_map ;
2020-09-28 02:52:18 +08:00
ASSERT ( buildConfiguration ( config , hack_map ) ! = ConfigurationResult : : NO_OPTIONS_PROVIDED ) ;
2017-08-22 13:29:56 +08:00
for ( auto kv : hack_map ) db . set ( kv . first , kv . second ) ;
}
2017-08-12 06:06:36 +08:00
2017-08-22 13:29:56 +08:00
StringRef StringRefOf ( const char * s ) {
return StringRef ( ( uint8_t * ) s , strlen ( s ) ) ;
2017-08-12 06:06:36 +08:00
}
2017-05-26 04:48:44 +08:00
2018-09-18 09:32:39 +08:00
void SimulationConfig : : generateNormalConfig ( int minimumReplication , int minimumRegions ) {
2017-08-22 13:29:56 +08:00
set_config ( " new " ) ;
2018-10-25 16:25:41 +08:00
const bool simple = false ; // Set true to simplify simulation configs for easier debugging
2019-02-07 03:04:39 +08:00
// generateMachineTeamTestConfig set up the number of servers per machine and the number of machines such that
// if we do not remove the surplus server and machine teams, the simulation test will report error.
// This is needed to make sure the number of server (and machine) teams is no larger than the desired number.
2019-02-15 07:59:20 +08:00
bool generateMachineTeamTestConfig = BUGGIFY_WITH_PROB ( 0.1 ) ? true : false ;
2019-05-11 05:01:52 +08:00
bool generateFearless = simple ? false : ( minimumRegions > 1 | | deterministicRandom ( ) - > random01 ( ) < 0.5 ) ;
datacenters = simple ? 1 : ( generateFearless ? ( minimumReplication > 0 | | deterministicRandom ( ) - > random01 ( ) < 0.5 ? 4 : 6 ) : deterministicRandom ( ) - > randomInt ( 1 , 4 ) ) ;
if ( deterministicRandom ( ) - > random01 ( ) < 0.25 ) db . desiredTLogCount = deterministicRandom ( ) - > randomInt ( 1 , 7 ) ;
2020-09-11 08:44:15 +08:00
if ( deterministicRandom ( ) - > random01 ( ) < 0.25 ) db . commitProxyCount = deterministicRandom ( ) - > randomInt ( 1 , 7 ) ;
2020-08-06 15:01:57 +08:00
if ( deterministicRandom ( ) - > random01 ( ) < 0.25 ) db . grvProxyCount = deterministicRandom ( ) - > randomInt ( 1 , 4 ) ;
2019-05-11 05:01:52 +08:00
if ( deterministicRandom ( ) - > random01 ( ) < 0.25 ) db . resolverCount = deterministicRandom ( ) - > randomInt ( 1 , 7 ) ;
2020-04-22 01:47:15 +08:00
int storage_engine_type = deterministicRandom ( ) - > randomInt ( 0 , 4 ) ;
2020-01-24 08:52:42 +08:00
switch ( storage_engine_type ) {
case 0 : {
TEST ( true ) ; // Simulated cluster using ssd storage engine
2017-08-22 13:29:56 +08:00
set_config ( " ssd " ) ;
2020-01-24 08:52:42 +08:00
break ;
}
case 1 : {
TEST ( true ) ; // Simulated cluster using default memory storage engine
2017-08-22 13:29:56 +08:00
set_config ( " memory " ) ;
2020-01-24 08:52:42 +08:00
break ;
2017-08-22 13:29:56 +08:00
}
2020-01-24 08:52:42 +08:00
case 2 : {
TEST ( true ) ; // Simulated cluster using radix-tree storage engine
set_config ( " memory-radixtree-beta " ) ;
break ;
}
2020-03-30 21:57:35 +08:00
case 3 : {
2020-11-11 05:01:07 +08:00
TEST ( true ) ; // Simulated cluster using redwood storage engine
2020-03-30 21:57:35 +08:00
set_config ( " ssd-redwood-experimental " ) ;
break ;
}
2020-01-24 08:52:42 +08:00
default :
ASSERT ( false ) ; // Programmer forgot to adjust cases.
}
2019-01-10 10:03:54 +08:00
// if (deterministicRandom()->random01() < 0.5) {
// set_config("ssd");
// } else {
// set_config("memory");
// }
// set_config("memory");
2020-01-25 08:40:11 +08:00
// set_config("memory-radixtree-beta");
2018-10-25 16:25:41 +08:00
if ( simple ) {
db . desiredTLogCount = 1 ;
2020-09-11 08:44:15 +08:00
db . commitProxyCount = 1 ;
2020-08-06 15:01:57 +08:00
db . grvProxyCount = 1 ;
2018-10-25 16:25:41 +08:00
db . resolverCount = 1 ;
}
2019-05-11 05:01:52 +08:00
int replication_type = simple ? 1 : ( std : : max ( minimumReplication , datacenters > 4 ? deterministicRandom ( ) - > randomInt ( 1 , 3 ) : std : : min ( deterministicRandom ( ) - > randomInt ( 0 , 6 ) , 3 ) ) ) ;
2017-08-22 13:29:56 +08:00
switch ( replication_type ) {
case 0 : {
2017-05-26 04:48:44 +08:00
TEST ( true ) ; // Simulated cluster using custom redundancy mode
2019-05-11 05:01:52 +08:00
int storage_servers = deterministicRandom ( ) - > randomInt ( 1 , generateFearless ? 4 : 5 ) ;
2018-06-11 11:22:58 +08:00
//FIXME: log replicas must be more than storage replicas because otherwise better master exists will not recognize it needs to change dcs
2019-05-11 05:01:52 +08:00
int replication_factor = deterministicRandom ( ) - > randomInt ( storage_servers , generateFearless ? 4 : 5 ) ;
int anti_quorum = deterministicRandom ( ) - > randomInt ( 0 , ( replication_factor / 2 ) + 1 ) ; //The anti quorum cannot be more than half of the replication factor, or the log system will continue to accept commits when a recovery is impossible
2017-08-22 13:29:56 +08:00
// Go through buildConfiguration, as it sets tLogPolicy/storagePolicy.
2018-06-19 01:24:57 +08:00
set_config ( format ( " storage_replicas:=%d log_replicas:=%d log_anti_quorum:=%d "
2017-08-22 13:29:56 +08:00
" replica_datacenters:=1 min_replica_datacenters:=1 " ,
2018-06-19 01:24:57 +08:00
storage_servers , replication_factor , anti_quorum ) ) ;
2017-08-22 13:29:56 +08:00
break ;
2017-05-26 04:48:44 +08:00
}
2017-08-12 06:06:36 +08:00
case 1 : {
2017-05-26 04:48:44 +08:00
TEST ( true ) ; // Simulated cluster running in single redundancy mode
2017-08-22 13:29:56 +08:00
set_config ( " single " ) ;
break ;
2017-05-26 04:48:44 +08:00
}
2017-08-12 06:06:36 +08:00
case 2 : {
2017-05-26 04:48:44 +08:00
TEST ( true ) ; // Simulated cluster running in double redundancy mode
2017-08-22 13:29:56 +08:00
set_config ( " double " ) ;
break ;
2017-05-26 04:48:44 +08:00
}
2017-08-12 06:06:36 +08:00
case 3 : {
2018-04-11 08:02:43 +08:00
if ( datacenters < = 2 | | generateFearless ) {
2017-05-26 04:48:44 +08:00
TEST ( true ) ; // Simulated cluster running in triple redundancy mode
2017-08-22 13:29:56 +08:00
set_config ( " triple " ) ;
2017-05-26 04:48:44 +08:00
}
2017-08-12 06:06:36 +08:00
else if ( datacenters = = 3 ) {
2017-05-26 04:48:44 +08:00
TEST ( true ) ; // Simulated cluster running in 3 data-hall mode
2017-08-22 13:29:56 +08:00
set_config ( " three_data_hall " ) ;
2017-05-26 04:48:44 +08:00
}
else {
ASSERT ( false ) ;
}
2017-08-22 13:29:56 +08:00
break ;
}
default :
ASSERT ( false ) ; // Programmer forgot to adjust cases.
2017-05-26 04:48:44 +08:00
}
2017-08-22 13:29:56 +08:00
2019-05-11 05:01:52 +08:00
if ( deterministicRandom ( ) - > random01 ( ) < 0.5 ) {
2019-07-09 13:22:45 +08:00
int logSpill = deterministicRandom ( ) - > randomInt ( TLogSpillType : : VALUE , TLogSpillType : : END ) ;
set_config ( format ( " log_spill:=%d " , logSpill ) ) ;
int logVersion = deterministicRandom ( ) - > randomInt ( TLogVersion : : MIN_RECRUITABLE , TLogVersion : : MAX_SUPPORTED + 1 ) ;
set_config ( format ( " log_version:=%d " , logVersion ) ) ;
2019-02-27 09:58:59 +08:00
} else {
2019-07-09 13:22:45 +08:00
if ( deterministicRandom ( ) - > random01 ( ) < 0.7 )
set_config ( format ( " log_version:=%d " , TLogVersion : : MAX_SUPPORTED ) ) ;
if ( deterministicRandom ( ) - > random01 ( ) < 0.5 )
set_config ( format ( " log_spill:=%d " , TLogSpillType : : DEFAULT ) ) ;
2019-02-20 14:02:44 +08:00
}
2020-02-06 06:23:54 +08:00
if ( deterministicRandom ( ) - > random01 ( ) < 0.5 ) {
set_config ( " backup_worker_enabled:=1 " ) ;
}
2019-05-11 05:01:52 +08:00
if ( generateFearless | | ( datacenters = = 2 & & deterministicRandom ( ) - > random01 ( ) < 0.5 ) ) {
2018-09-18 09:32:39 +08:00
//The kill region workload relies on the fact that all "0", "2", and "4" are all of the possible primary dcids.
2018-03-06 11:27:46 +08:00
StatusObject primaryObj ;
2018-06-13 07:18:54 +08:00
StatusObject primaryDcObj ;
primaryDcObj [ " id " ] = " 0 " ;
2018-06-14 09:35:28 +08:00
primaryDcObj [ " priority " ] = 2 ;
2018-06-13 07:18:54 +08:00
StatusArray primaryDcArr ;
primaryDcArr . push_back ( primaryDcObj ) ;
2018-03-06 11:27:46 +08:00
StatusObject remoteObj ;
2018-06-13 07:18:54 +08:00
StatusObject remoteDcObj ;
remoteDcObj [ " id " ] = " 1 " ;
remoteDcObj [ " priority " ] = 1 ;
StatusArray remoteDcArr ;
remoteDcArr . push_back ( remoteDcObj ) ;
2018-03-16 06:40:58 +08:00
bool needsRemote = generateFearless ;
2018-03-06 11:27:46 +08:00
if ( generateFearless ) {
2018-07-10 13:01:46 +08:00
if ( datacenters > 4 ) {
//FIXME: we cannot use one satellite replication with more than one satellite per region because canKillProcesses does not respect usable_dcs
2019-05-11 05:01:52 +08:00
int satellite_replication_type = deterministicRandom ( ) - > randomInt ( 0 , 3 ) ;
2018-07-10 13:01:46 +08:00
switch ( satellite_replication_type ) {
case 0 : {
2020-11-11 05:01:07 +08:00
TEST ( true ) ; // Simulated cluster using no satellite redundancy mode (>4 datacenters)
2018-07-10 13:01:46 +08:00
break ;
}
case 1 : {
TEST ( true ) ; // Simulated cluster using two satellite fast redundancy mode
primaryObj [ " satellite_redundancy_mode " ] = " two_satellite_fast " ;
remoteObj [ " satellite_redundancy_mode " ] = " two_satellite_fast " ;
break ;
}
case 2 : {
TEST ( true ) ; // Simulated cluster using two satellite safe redundancy mode
primaryObj [ " satellite_redundancy_mode " ] = " two_satellite_safe " ;
remoteObj [ " satellite_redundancy_mode " ] = " two_satellite_safe " ;
break ;
}
default :
ASSERT ( false ) ; // Programmer forgot to adjust cases.
}
} else {
2019-05-11 05:01:52 +08:00
int satellite_replication_type = deterministicRandom ( ) - > randomInt ( 0 , 5 ) ;
2018-07-10 13:01:46 +08:00
switch ( satellite_replication_type ) {
case 0 : {
//FIXME: implement
TEST ( true ) ; // Simulated cluster using custom satellite redundancy mode
break ;
}
case 1 : {
2020-11-11 05:01:07 +08:00
TEST ( true ) ; // Simulated cluster using no satellite redundancy mode (<4 datacenters)
2018-07-10 13:01:46 +08:00
break ;
}
case 2 : {
TEST ( true ) ; // Simulated cluster using single satellite redundancy mode
primaryObj [ " satellite_redundancy_mode " ] = " one_satellite_single " ;
remoteObj [ " satellite_redundancy_mode " ] = " one_satellite_single " ;
break ;
}
case 3 : {
TEST ( true ) ; // Simulated cluster using double satellite redundancy mode
primaryObj [ " satellite_redundancy_mode " ] = " one_satellite_double " ;
remoteObj [ " satellite_redundancy_mode " ] = " one_satellite_double " ;
break ;
}
case 4 : {
TEST ( true ) ; // Simulated cluster using triple satellite redundancy mode
primaryObj [ " satellite_redundancy_mode " ] = " one_satellite_triple " ;
remoteObj [ " satellite_redundancy_mode " ] = " one_satellite_triple " ;
break ;
}
default :
ASSERT ( false ) ; // Programmer forgot to adjust cases.
}
2018-03-06 11:27:46 +08:00
}
2018-02-19 04:59:43 +08:00
2019-10-15 09:30:15 +08:00
if ( deterministicRandom ( ) - > random01 ( ) < 0.25 ) primaryObj [ " satellite_logs " ] = deterministicRandom ( ) - > randomInt ( 1 , 7 ) ;
if ( deterministicRandom ( ) - > random01 ( ) < 0.25 ) remoteObj [ " satellite_logs " ] = deterministicRandom ( ) - > randomInt ( 1 , 7 ) ;
2018-06-14 08:55:55 +08:00
2018-07-10 07:55:33 +08:00
//We cannot run with a remote DC when MAX_READ_TRANSACTION_LIFE_VERSIONS is too small, because the log routers will not be able to keep up.
2019-05-11 05:01:52 +08:00
if ( minimumRegions < = 1 & & ( deterministicRandom ( ) - > random01 ( ) < 0.25 | | SERVER_KNOBS - > MAX_READ_TRANSACTION_LIFE_VERSIONS < SERVER_KNOBS - > VERSIONS_PER_SECOND ) ) {
2018-06-18 10:31:15 +08:00
TEST ( true ) ; // Simulated cluster using one region
needsRemote = false ;
} else {
TEST ( true ) ; // Simulated cluster using two regions
db . usableRegions = 2 ;
}
2019-05-11 05:01:52 +08:00
int remote_replication_type = deterministicRandom ( ) - > randomInt ( 0 , datacenters > 4 ? 4 : 5 ) ;
2018-03-06 11:27:46 +08:00
switch ( remote_replication_type ) {
case 0 : {
2018-03-16 01:59:30 +08:00
//FIXME: implement
2018-03-06 11:27:46 +08:00
TEST ( true ) ; // Simulated cluster using custom remote redundancy mode
break ;
}
case 1 : {
2018-06-18 10:31:15 +08:00
TEST ( true ) ; // Simulated cluster using default remote redundancy mode
2018-03-06 11:27:46 +08:00
break ;
}
case 2 : {
TEST ( true ) ; // Simulated cluster using single remote redundancy mode
set_config ( " remote_single " ) ;
break ;
}
case 3 : {
TEST ( true ) ; // Simulated cluster using double remote redundancy mode
set_config ( " remote_double " ) ;
break ;
}
case 4 : {
TEST ( true ) ; // Simulated cluster using triple remote redundancy mode
set_config ( " remote_triple " ) ;
break ;
}
default :
ASSERT ( false ) ; // Programmer forgot to adjust cases.
}
2018-06-19 01:22:34 +08:00
2019-05-11 05:01:52 +08:00
if ( deterministicRandom ( ) - > random01 ( ) < 0.25 ) db . desiredLogRouterCount = deterministicRandom ( ) - > randomInt ( 1 , 7 ) ;
if ( deterministicRandom ( ) - > random01 ( ) < 0.25 ) db . remoteDesiredTLogCount = deterministicRandom ( ) - > randomInt ( 1 , 7 ) ;
2019-03-19 03:17:59 +08:00
2019-05-11 05:01:52 +08:00
bool useNormalDCsAsSatellites = datacenters > 4 & & minimumRegions < 2 & & deterministicRandom ( ) - > random01 ( ) < 0.3 ;
2019-03-19 03:17:59 +08:00
StatusObject primarySatelliteObj ;
primarySatelliteObj [ " id " ] = useNormalDCsAsSatellites ? " 1 " : " 2 " ;
primarySatelliteObj [ " priority " ] = 1 ;
primarySatelliteObj [ " satellite " ] = 1 ;
2019-10-15 09:30:15 +08:00
if ( deterministicRandom ( ) - > random01 ( ) < 0.25 ) primarySatelliteObj [ " satellite_logs " ] = deterministicRandom ( ) - > randomInt ( 1 , 7 ) ;
2019-03-19 03:17:59 +08:00
primaryDcArr . push_back ( primarySatelliteObj ) ;
StatusObject remoteSatelliteObj ;
remoteSatelliteObj [ " id " ] = useNormalDCsAsSatellites ? " 0 " : " 3 " ;
remoteSatelliteObj [ " priority " ] = 1 ;
remoteSatelliteObj [ " satellite " ] = 1 ;
2019-10-15 09:30:15 +08:00
if ( deterministicRandom ( ) - > random01 ( ) < 0.25 ) remoteSatelliteObj [ " satellite_logs " ] = deterministicRandom ( ) - > randomInt ( 1 , 7 ) ;
2019-03-19 03:17:59 +08:00
remoteDcArr . push_back ( remoteSatelliteObj ) ;
if ( datacenters > 4 ) {
StatusObject primarySatelliteObjB ;
primarySatelliteObjB [ " id " ] = useNormalDCsAsSatellites ? " 2 " : " 4 " ;
primarySatelliteObjB [ " priority " ] = 1 ;
primarySatelliteObjB [ " satellite " ] = 1 ;
2019-10-15 09:30:15 +08:00
if ( deterministicRandom ( ) - > random01 ( ) < 0.25 ) primarySatelliteObjB [ " satellite_logs " ] = deterministicRandom ( ) - > randomInt ( 1 , 7 ) ;
2019-03-19 03:17:59 +08:00
primaryDcArr . push_back ( primarySatelliteObjB ) ;
StatusObject remoteSatelliteObjB ;
remoteSatelliteObjB [ " id " ] = useNormalDCsAsSatellites ? " 2 " : " 5 " ;
remoteSatelliteObjB [ " priority " ] = 1 ;
remoteSatelliteObjB [ " satellite " ] = 1 ;
2019-10-15 09:30:15 +08:00
if ( deterministicRandom ( ) - > random01 ( ) < 0.25 ) remoteSatelliteObjB [ " satellite_logs " ] = deterministicRandom ( ) - > randomInt ( 1 , 7 ) ;
2019-03-19 03:17:59 +08:00
remoteDcArr . push_back ( remoteSatelliteObjB ) ;
}
if ( useNormalDCsAsSatellites ) {
datacenters = 3 ;
}
2018-03-06 11:27:46 +08:00
}
2018-06-14 08:55:55 +08:00
primaryObj [ " datacenters " ] = primaryDcArr ;
remoteObj [ " datacenters " ] = remoteDcArr ;
2018-03-06 11:27:46 +08:00
StatusArray regionArr ;
regionArr . push_back ( primaryObj ) ;
2019-05-11 05:01:52 +08:00
if ( needsRemote | | deterministicRandom ( ) - > random01 ( ) < 0.5 ) {
2018-03-16 06:40:58 +08:00
regionArr . push_back ( remoteObj ) ;
}
2018-03-06 11:27:46 +08:00
set_config ( " regions= " + json_spirit : : write_string ( json_spirit : : mValue ( regionArr ) , json_spirit : : Output_options : : none ) ) ;
2018-07-17 01:06:57 +08:00
if ( needsRemote ) {
2018-09-18 09:32:39 +08:00
g_simulator . originalRegions = " regions= " + json_spirit : : write_string ( json_spirit : : mValue ( regionArr ) , json_spirit : : Output_options : : none ) ;
2018-07-17 01:06:57 +08:00
StatusArray disablePrimary = regionArr ;
disablePrimary [ 0 ] . get_obj ( ) [ " datacenters " ] . get_array ( ) [ 0 ] . get_obj ( ) [ " priority " ] = - 1 ;
g_simulator . disablePrimary = " regions= " + json_spirit : : write_string ( json_spirit : : mValue ( disablePrimary ) , json_spirit : : Output_options : : none ) ;
StatusArray disableRemote = regionArr ;
disableRemote [ 1 ] . get_obj ( ) [ " datacenters " ] . get_array ( ) [ 0 ] . get_obj ( ) [ " priority " ] = - 1 ;
g_simulator . disableRemote = " regions= " + json_spirit : : write_string ( json_spirit : : mValue ( disableRemote ) , json_spirit : : Output_options : : none ) ;
}
2018-02-19 04:59:43 +08:00
}
2018-06-14 08:55:55 +08:00
if ( generateFearless & & minimumReplication > 1 ) {
2018-05-08 02:28:25 +08:00
//low latency tests in fearless configurations need 4 machines per datacenter (3 for triple replication, 1 that is down during failures).
2020-08-08 02:44:01 +08:00
machine_count = 16 ;
2018-05-08 02:28:25 +08:00
} else if ( generateFearless ) {
2020-08-17 06:10:36 +08:00
machine_count = 12 ;
2018-02-19 04:59:43 +08:00
} else if ( db . tLogPolicy & & db . tLogPolicy - > info ( ) = = " data_hall^2 x zoneid^2 x 1 " ) {
2020-08-08 02:44:01 +08:00
machine_count = 9 ;
2018-02-11 05:27:51 +08:00
} else {
2018-02-16 10:32:39 +08:00
//datacenters+2 so that the configure database workload can configure into three_data_hall
2019-07-31 05:02:31 +08:00
machine_count = std : : max ( datacenters + 2 , ( ( db . minDatacentersRequired ( ) > 0 ) ? datacenters : 1 ) * std : : max ( 3 , db . minZonesRequiredPerDatacenter ( ) ) ) ;
2019-05-11 05:01:52 +08:00
machine_count = deterministicRandom ( ) - > randomInt ( machine_count , std : : max ( machine_count + 1 , extraDB ? 6 : 10 ) ) ;
2019-02-16 07:10:58 +08:00
if ( generateMachineTeamTestConfig ) {
// When DESIRED_TEAMS_PER_SERVER is set to 1, the desired machine team number is 5
// while the max possible machine team number is 10.
// If machine_count > 5, we can still test the effectivenss of machine teams
// Note: machine_count may be much larger than 5 because we may have a big replication factor
2019-05-11 05:01:52 +08:00
machine_count = std : : max ( machine_count , deterministicRandom ( ) - > randomInt ( 5 , extraDB ? 6 : 10 ) ) ;
2019-02-16 07:10:58 +08:00
}
2018-02-11 05:27:51 +08:00
}
2018-02-16 10:32:39 +08:00
//because we protect a majority of coordinators from being killed, it is better to run with low numbers of coordinators to prevent too many processes from being protected
2019-05-11 05:01:52 +08:00
coordinators = ( minimumRegions < = 1 & & BUGGIFY ) ? deterministicRandom ( ) - > randomInt ( 1 , std : : max ( machine_count , 2 ) ) : 1 ;
2018-02-16 10:32:39 +08:00
2017-11-29 10:20:29 +08:00
if ( minimumReplication > 1 & & datacenters = = 3 ) {
//low latency tests in 3 data hall mode need 2 other data centers with 2 machines each to avoid waiting for logs to recover.
machine_count = std : : max ( machine_count , 6 ) ;
2018-02-16 10:32:39 +08:00
coordinators = 3 ;
2017-11-29 10:20:29 +08:00
}
2018-02-19 04:59:43 +08:00
if ( generateFearless ) {
processes_per_machine = 1 ;
} else {
2019-05-11 05:01:52 +08:00
processes_per_machine = deterministicRandom ( ) - > randomInt ( 1 , ( extraDB ? 14 : 28 ) / machine_count + 2 ) ;
2018-02-19 04:59:43 +08:00
}
2017-08-12 06:06:36 +08:00
}
2017-05-26 04:48:44 +08:00
2019-02-28 16:09:53 +08:00
void setupSimulatedSystem ( vector < Future < Void > > * systemActors , std : : string baseFolder , int * pTesterCount ,
Optional < ClusterConnectionString > * pConnString , Standalone < StringRef > * pStartingConfiguration ,
2020-10-07 02:33:32 +08:00
int extraDB , int minimumReplication , int minimumRegions , std : : string whitelistBinPaths ,
bool configureLocked , int logAntiQuorum ) {
2017-05-26 04:48:44 +08:00
// SOMEDAY: this does not test multi-interface configurations
2018-09-18 09:32:39 +08:00
SimulationConfig simconfig ( extraDB , minimumReplication , minimumRegions ) ;
2020-10-07 02:33:32 +08:00
if ( logAntiQuorum ! = - 1 ) {
simconfig . db . tLogWriteAntiQuorum = logAntiQuorum ;
}
2018-03-06 11:27:46 +08:00
StatusObject startingConfigJSON = simconfig . db . toJSON ( true ) ;
std : : string startingConfigString = " new " ;
2020-02-12 03:10:09 +08:00
if ( configureLocked ) {
startingConfigString + = " locked " ;
}
2018-03-06 11:27:46 +08:00
for ( auto kv : startingConfigJSON ) {
startingConfigString + = " " ;
if ( kv . second . type ( ) = = json_spirit : : int_type ) {
2019-01-19 07:42:48 +08:00
startingConfigString + = kv . first + " := " + format ( " %d " , kv . second . get_int ( ) ) ;
2018-03-06 11:27:46 +08:00
} else if ( kv . second . type ( ) = = json_spirit : : str_type ) {
2019-01-19 07:42:48 +08:00
startingConfigString + = kv . second . get_str ( ) ;
2018-03-06 11:27:46 +08:00
} else if ( kv . second . type ( ) = = json_spirit : : array_type ) {
2019-01-19 07:42:48 +08:00
startingConfigString + = kv . first + " = " + json_spirit : : write_string ( json_spirit : : mValue ( kv . second . get_array ( ) ) , json_spirit : : Output_options : : none ) ;
2018-03-06 11:27:46 +08:00
} else {
ASSERT ( false ) ;
}
}
2017-08-22 13:29:56 +08:00
g_simulator . storagePolicy = simconfig . db . storagePolicy ;
g_simulator . tLogPolicy = simconfig . db . tLogPolicy ;
g_simulator . tLogWriteAntiQuorum = simconfig . db . tLogWriteAntiQuorum ;
2018-06-18 10:31:15 +08:00
g_simulator . remoteTLogPolicy = simconfig . db . getRemoteTLogPolicy ( ) ;
g_simulator . usableRegions = simconfig . db . usableRegions ;
2017-10-20 06:49:54 +08:00
2018-06-29 14:15:32 +08:00
if ( simconfig . db . regions . size ( ) > 0 ) {
2018-03-06 11:27:46 +08:00
g_simulator . primaryDcId = simconfig . db . regions [ 0 ] . dcId ;
2018-06-11 11:20:41 +08:00
g_simulator . hasSatelliteReplication = simconfig . db . regions [ 0 ] . satelliteTLogReplicationFactor > 0 ;
2018-06-29 14:15:32 +08:00
if ( simconfig . db . regions [ 0 ] . satelliteTLogUsableDcsFallback > 0 ) {
2018-07-10 13:01:46 +08:00
g_simulator . satelliteTLogPolicyFallback = simconfig . db . regions [ 0 ] . satelliteTLogPolicyFallback ;
g_simulator . satelliteTLogWriteAntiQuorumFallback = simconfig . db . regions [ 0 ] . satelliteTLogWriteAntiQuorumFallback ;
2018-06-29 14:15:32 +08:00
} else {
2018-07-10 13:01:46 +08:00
g_simulator . satelliteTLogPolicyFallback = simconfig . db . regions [ 0 ] . satelliteTLogPolicy ;
g_simulator . satelliteTLogWriteAntiQuorumFallback = simconfig . db . regions [ 0 ] . satelliteTLogWriteAntiQuorum ;
2018-03-06 11:27:46 +08:00
}
2018-07-10 13:01:46 +08:00
g_simulator . satelliteTLogPolicy = simconfig . db . regions [ 0 ] . satelliteTLogPolicy ;
g_simulator . satelliteTLogWriteAntiQuorum = simconfig . db . regions [ 0 ] . satelliteTLogWriteAntiQuorum ;
2018-03-16 01:59:30 +08:00
for ( auto s : simconfig . db . regions [ 0 ] . satellites ) {
g_simulator . primarySatelliteDcIds . push_back ( s . dcId ) ;
}
2018-03-06 11:27:46 +08:00
} else {
g_simulator . hasSatelliteReplication = false ;
g_simulator . satelliteTLogWriteAntiQuorum = 0 ;
}
2018-06-29 14:15:32 +08:00
if ( simconfig . db . regions . size ( ) = = 2 ) {
g_simulator . remoteDcId = simconfig . db . regions [ 1 ] . dcId ;
ASSERT ( ( ! simconfig . db . regions [ 0 ] . satelliteTLogPolicy & & ! simconfig . db . regions [ 1 ] . satelliteTLogPolicy ) | | simconfig . db . regions [ 0 ] . satelliteTLogPolicy - > info ( ) = = simconfig . db . regions [ 1 ] . satelliteTLogPolicy - > info ( ) ) ;
for ( auto s : simconfig . db . regions [ 1 ] . satellites ) {
g_simulator . remoteSatelliteDcIds . push_back ( s . dcId ) ;
}
}
2018-07-10 13:24:43 +08:00
if ( g_simulator . usableRegions < 2 | | ! g_simulator . hasSatelliteReplication ) {
g_simulator . allowLogSetKills = false ;
}
2017-10-20 06:49:54 +08:00
ASSERT ( g_simulator . storagePolicy & & g_simulator . tLogPolicy ) ;
ASSERT ( ! g_simulator . hasSatelliteReplication | | g_simulator . satelliteTLogPolicy ) ;
2019-03-19 06:03:43 +08:00
TraceEvent ( " SimulatorConfig " ) . detail ( " ConfigString " , StringRef ( startingConfigString ) ) ;
2017-05-26 04:48:44 +08:00
2017-08-22 13:29:56 +08:00
const int dataCenters = simconfig . datacenters ;
const int machineCount = simconfig . machine_count ;
const int coordinatorCount = simconfig . coordinators ;
const int processesPerMachine = simconfig . processes_per_machine ;
2017-05-26 04:48:44 +08:00
// half the time, when we have more than 4 machines that are not the first in their dataCenter, assign classes
2019-05-11 05:01:52 +08:00
bool assignClasses = machineCount - dataCenters > 4 & & deterministicRandom ( ) - > random01 ( ) < 0.5 ;
2017-05-26 04:48:44 +08:00
2017-10-30 23:31:01 +08:00
// Use SSL 5% of the time
2020-02-07 11:29:31 +08:00
bool sslEnabled = deterministicRandom ( ) - > random01 ( ) < 0.10 ;
2019-05-11 05:01:52 +08:00
bool sslOnly = sslEnabled & & deterministicRandom ( ) - > coinflip ( ) ;
2019-02-01 10:20:14 +08:00
g_simulator . listenersPerProcess = sslEnabled & & ! sslOnly ? 2 : 1 ;
2017-05-26 04:48:44 +08:00
TEST ( sslEnabled ) ; // SSL enabled
TEST ( ! sslEnabled ) ; // SSL disabled
2019-02-28 16:09:53 +08:00
// Use IPv6 25% of the time
2019-05-11 05:01:52 +08:00
bool useIPv6 = deterministicRandom ( ) - > random01 ( ) < 0.25 ;
2020-11-11 05:01:07 +08:00
TEST ( useIPv6 ) ; // Use IPv6
TEST ( ! useIPv6 ) ; // Use IPv4
2019-02-28 16:09:53 +08:00
2017-05-26 04:48:44 +08:00
vector < NetworkAddress > coordinatorAddresses ;
2018-09-18 09:32:39 +08:00
if ( minimumRegions > 1 ) {
//do not put coordinators in the primary region so that we can kill that region safely
int nonPrimaryDcs = dataCenters / 2 ;
for ( int dc = 1 ; dc < dataCenters ; dc + = 2 ) {
int dcCoordinators = coordinatorCount / nonPrimaryDcs + ( ( dc - 1 ) / 2 < coordinatorCount % nonPrimaryDcs ) ;
for ( int m = 0 ; m < dcCoordinators ; m + + ) {
2019-02-28 16:09:53 +08:00
auto ip = makeIPAddressForSim ( useIPv6 , { 2 , dc , 1 , m } ) ;
2019-02-01 10:20:14 +08:00
coordinatorAddresses . push_back ( NetworkAddress ( ip , sslEnabled & & ! sslOnly ? 2 : 1 , true , sslEnabled & & sslOnly ) ) ;
2018-09-18 09:32:39 +08:00
TraceEvent ( " SelectedCoordinator " ) . detail ( " Address " , coordinatorAddresses . back ( ) ) ;
}
}
} else {
2019-01-19 07:42:48 +08:00
int assignedMachines = 0 ;
int coordCount = coordinatorCount ;
if ( coordinatorCount > 4 ) {
+ + coordCount ;
}
2018-09-18 09:32:39 +08:00
for ( int dc = 0 ; dc < dataCenters ; dc + + ) {
2019-01-19 07:42:48 +08:00
int dcCoordinators = coordCount / dataCenters + ( dc < coordCount % dataCenters ) ;
int machines = machineCount / dataCenters + ( dc < machineCount % dataCenters ) ;
2018-09-18 09:32:39 +08:00
for ( int m = 0 ; m < dcCoordinators ; m + + ) {
2019-01-19 07:42:48 +08:00
if ( coordinatorCount > 4 & & ( assignedMachines = = 4 | | ( m + 1 = = dcCoordinators & & assignedMachines < 4 & & assignedMachines + machines - dcCoordinators > = 4 ) ) ) {
2019-02-28 16:09:53 +08:00
auto ip = makeIPAddressForSim ( useIPv6 , { 2 , dc , 1 , m } ) ;
TraceEvent ( " SkippedCoordinator " )
. detail ( " Address " , ip . toString ( ) )
. detail ( " M " , m )
. detail ( " Machines " , machines )
. detail ( " Assigned " , assignedMachines )
. detail ( " DcCoord " , dcCoordinators )
. detail ( " CoordinatorCount " , coordinatorCount ) ;
2019-01-19 07:42:48 +08:00
} else {
2019-02-28 16:09:53 +08:00
auto ip = makeIPAddressForSim ( useIPv6 , { 2 , dc , 1 , m } ) ;
2019-02-13 17:52:59 +08:00
coordinatorAddresses . push_back ( NetworkAddress ( ip , sslEnabled & & ! sslOnly ? 2 : 1 , true , sslEnabled & & sslOnly ) ) ;
2019-01-19 07:42:48 +08:00
TraceEvent ( " SelectedCoordinator " ) . detail ( " Address " , coordinatorAddresses . back ( ) ) . detail ( " M " , m ) . detail ( " Machines " , machines ) . detail ( " Assigned " , assignedMachines ) . detail ( " DcCoord " , dcCoordinators ) . detail ( " P1 " , ( m + 1 = = dcCoordinators ) ) . detail ( " P2 " , ( assignedMachines < 4 ) ) . detail ( " P3 " , ( assignedMachines + machines - dcCoordinators > = 4 ) ) . detail ( " CoordinatorCount " , coordinatorCount ) ;
}
assignedMachines + + ;
2018-09-18 09:32:39 +08:00
}
2019-01-19 07:42:48 +08:00
assignedMachines + = machines - dcCoordinators ;
2017-05-26 04:48:44 +08:00
}
}
2017-05-27 05:20:11 +08:00
2019-05-11 05:01:52 +08:00
deterministicRandom ( ) - > randomShuffle ( coordinatorAddresses ) ;
2017-05-26 04:48:44 +08:00
for ( int i = 0 ; i < ( coordinatorAddresses . size ( ) / 2 ) + 1 ; i + + ) {
2019-02-28 16:09:53 +08:00
TraceEvent ( " ProtectCoordinator " )
. detail ( " Address " , coordinatorAddresses [ i ] )
. detail ( " Coordinators " , describe ( coordinatorAddresses ) ) ;
g_simulator . protectedAddresses . insert (
2020-02-12 07:52:06 +08:00
NetworkAddress ( coordinatorAddresses [ i ] . ip , coordinatorAddresses [ i ] . port , true , coordinatorAddresses [ i ] . isTLS ( ) ) ) ;
2019-02-01 10:20:14 +08:00
if ( coordinatorAddresses [ i ] . port = = 2 ) {
2020-02-13 07:12:43 +08:00
g_simulator . protectedAddresses . insert ( NetworkAddress ( coordinatorAddresses [ i ] . ip , 1 , true , true ) ) ;
2019-02-01 10:20:14 +08:00
}
2017-05-26 04:48:44 +08:00
}
2019-05-11 05:01:52 +08:00
deterministicRandom ( ) - > randomShuffle ( coordinatorAddresses ) ;
2017-05-27 05:20:11 +08:00
2020-07-21 05:13:52 +08:00
ASSERT_EQ ( coordinatorAddresses . size ( ) , coordinatorCount ) ;
2017-05-26 04:48:44 +08:00
ClusterConnectionString conn ( coordinatorAddresses , LiteralStringRef ( " TestCluster:0 " ) ) ;
2018-04-27 08:24:40 +08:00
2018-04-19 10:34:35 +08:00
// If extraDB==0, leave g_simulator.extraDB as null because the test does not use DR.
if ( extraDB = = 1 ) {
// The DR database can be either a new database or itself
g_simulator . extraDB = new ClusterConnectionString ( coordinatorAddresses , BUGGIFY ? LiteralStringRef ( " TestCluster:0 " ) : LiteralStringRef ( " ExtraCluster:0 " ) ) ;
} else if ( extraDB = = 2 ) {
// The DR database is a new database
g_simulator . extraDB = new ClusterConnectionString ( coordinatorAddresses , LiteralStringRef ( " ExtraCluster:0 " ) ) ;
} else if ( extraDB = = 3 ) {
// The DR database is the same database
g_simulator . extraDB = new ClusterConnectionString ( coordinatorAddresses , LiteralStringRef ( " TestCluster:0 " ) ) ;
}
2017-05-26 04:48:44 +08:00
* pConnString = conn ;
2019-04-04 09:03:12 +08:00
TraceEvent ( " SimulatedConnectionString " ) . detail ( " String " , conn . toString ( ) ) . detail ( " ConfigString " , startingConfigString ) ;
2017-05-26 04:48:44 +08:00
2019-07-24 10:24:07 +08:00
bool requiresExtraDBMachines = extraDB & & g_simulator . extraDB - > toString ( ) ! = conn . toString ( ) ;
2017-05-26 04:48:44 +08:00
int assignedMachines = 0 , nonVersatileMachines = 0 ;
2019-11-13 06:22:36 +08:00
std : : vector < ProcessClass : : ClassType > processClassesSubSet = { ProcessClass : : UnsetClass , ProcessClass : : ResolutionClass , ProcessClass : : MasterClass } ;
2017-05-26 04:48:44 +08:00
for ( int dc = 0 ; dc < dataCenters ; dc + + ) {
2018-02-11 05:27:51 +08:00
//FIXME: test unset dcID
2017-10-20 06:49:54 +08:00
Optional < Standalone < StringRef > > dcUID = StringRef ( format ( " %d " , dc ) ) ;
2017-05-26 04:48:44 +08:00
std : : vector < UID > machineIdentities ;
int machines = machineCount / dataCenters + ( dc < machineCount % dataCenters ) ; // add remainder of machines to first datacenter
int dcCoordinators = coordinatorCount / dataCenters + ( dc < coordinatorCount % dataCenters ) ;
printf ( " Datacenter %d: %d/%d machines, %d/%d coordinators \n " , dc , machines , machineCount , dcCoordinators , coordinatorCount ) ;
2020-07-21 05:13:52 +08:00
ASSERT_LE ( dcCoordinators , machines ) ;
2020-01-24 08:52:42 +08:00
2019-11-13 05:01:29 +08:00
//FIXME: temporarily code to test storage cache
//TODO: caching disabled for this merge
2019-12-06 09:10:32 +08:00
if ( dc = = 0 ) {
machines + + ;
}
2019-11-13 05:01:29 +08:00
2019-05-11 05:01:52 +08:00
int useSeedForMachine = deterministicRandom ( ) - > randomInt ( 0 , machines ) ;
2019-01-19 07:42:48 +08:00
Standalone < StringRef > zoneId ;
Standalone < StringRef > newZoneId ;
2017-05-26 04:48:44 +08:00
for ( int machine = 0 ; machine < machines ; machine + + ) {
2019-05-11 05:01:52 +08:00
Standalone < StringRef > machineId ( deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ) ;
if ( machine = = 0 | | machineCount - dataCenters < = 4 | | assignedMachines ! = 4 | | simconfig . db . regions . size ( ) | | deterministicRandom ( ) - > random01 ( ) < 0.5 ) {
zoneId = deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ;
newZoneId = deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ;
2019-01-19 07:42:48 +08:00
}
2017-05-26 04:48:44 +08:00
//Choose a machine class
ProcessClass processClass = ProcessClass ( ProcessClass : : UnsetClass , ProcessClass : : CommandLineSource ) ;
if ( assignClasses ) {
if ( assignedMachines < 4 )
2019-05-11 05:01:52 +08:00
processClass = ProcessClass ( ( ProcessClass : : ClassType ) deterministicRandom ( ) - > randomInt ( 0 , 2 ) , ProcessClass : : CommandLineSource ) ; //Unset or Storage
2018-04-12 12:22:53 +08:00
else if ( assignedMachines = = 4 & & ! simconfig . db . regions . size ( ) )
2019-11-13 06:22:36 +08:00
processClass = ProcessClass ( processClassesSubSet [ deterministicRandom ( ) - > randomInt ( 0 , processClassesSubSet . size ( ) ) ] , ProcessClass : : CommandLineSource ) ; //Unset or Resolution or Master
2017-05-26 04:48:44 +08:00
else
2019-05-11 05:01:52 +08:00
processClass = ProcessClass ( ( ProcessClass : : ClassType ) deterministicRandom ( ) - > randomInt ( 0 , 3 ) , ProcessClass : : CommandLineSource ) ; //Unset, Storage, or Transaction
2017-05-26 04:48:44 +08:00
if ( processClass = = ProcessClass : : ResolutionClass ) // *can't* be assigned to other roles, even in an emergency
nonVersatileMachines + + ;
}
2019-11-13 05:01:29 +08:00
//FIXME: temporarily code to test storage cache
//TODO: caching disabled for this merge
2019-12-06 09:10:32 +08:00
if ( machine = = machines - 1 & & dc = = 0 ) {
processClass = ProcessClass ( ProcessClass : : StorageCacheClass , ProcessClass : : CommandLineSource ) ;
nonVersatileMachines + + ;
}
2019-11-13 05:01:29 +08:00
2019-02-28 16:09:53 +08:00
std : : vector < IPAddress > ips ;
for ( int i = 0 ; i < processesPerMachine ; i + + ) {
2019-05-11 05:01:52 +08:00
ips . push_back ( makeIPAddressForSim ( useIPv6 , { 2 , dc , deterministicRandom ( ) - > randomInt ( 1 , i + 2 ) , machine } ) ) ;
2017-05-26 04:48:44 +08:00
}
2019-07-24 10:24:07 +08:00
if ( requiresExtraDBMachines ) {
ips . push_back ( makeIPAddressForSim ( useIPv6 , { 2 , dc , 1 , machine } ) ) ;
}
2020-07-13 09:30:02 +08:00
// check the sslEnablementMap using only one ip
2019-01-19 07:42:48 +08:00
LocalityData localities ( Optional < Standalone < StringRef > > ( ) , zoneId , machineId , dcUID ) ;
2017-05-26 04:48:44 +08:00
localities . set ( LiteralStringRef ( " data_hall " ) , dcUID ) ;
2020-02-07 11:29:31 +08:00
systemActors - > push_back ( reportErrors ( simulatedMachine ( conn , ips , sslEnabled ,
2019-07-24 10:24:07 +08:00
localities , processClass , baseFolder , false , machine = = useSeedForMachine , requiresExtraDBMachines ? AgentOnly : AgentAddition , sslOnly , whitelistBinPaths ) , " SimulatedMachine " ) ) ;
2017-05-26 04:48:44 +08:00
2019-07-24 10:24:07 +08:00
if ( requiresExtraDBMachines ) {
2019-02-28 16:09:53 +08:00
std : : vector < IPAddress > extraIps ;
2017-05-26 04:48:44 +08:00
for ( int i = 0 ; i < processesPerMachine ; i + + ) {
2019-05-11 05:01:52 +08:00
extraIps . push_back ( makeIPAddressForSim ( useIPv6 , { 4 , dc , deterministicRandom ( ) - > randomInt ( 1 , i + 2 ) , machine } ) ) ;
2017-05-26 04:48:44 +08:00
}
2019-05-11 05:01:52 +08:00
Standalone < StringRef > newMachineId ( deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ) ;
2019-01-19 07:42:48 +08:00
LocalityData localities ( Optional < Standalone < StringRef > > ( ) , newZoneId , newMachineId , dcUID ) ;
2017-05-26 04:48:44 +08:00
localities . set ( LiteralStringRef ( " data_hall " ) , dcUID ) ;
2020-02-07 11:29:31 +08:00
systemActors - > push_back ( reportErrors ( simulatedMachine ( * g_simulator . extraDB , extraIps , sslEnabled ,
2017-05-26 04:48:44 +08:00
localities ,
2019-07-24 10:24:07 +08:00
processClass , baseFolder , false , machine = = useSeedForMachine , AgentNone , sslOnly , whitelistBinPaths ) , " SimulatedMachine " ) ) ;
2017-05-26 04:48:44 +08:00
}
assignedMachines + + ;
}
}
g_simulator . desiredCoordinators = coordinatorCount ;
g_simulator . physicalDatacenters = dataCenters ;
g_simulator . processesPerMachine = processesPerMachine ;
TraceEvent ( " SetupSimulatorSettings " )
2018-06-09 02:11:08 +08:00
. detail ( " DesiredCoordinators " , g_simulator . desiredCoordinators )
. detail ( " PhysicalDatacenters " , g_simulator . physicalDatacenters )
. detail ( " ProcessesPerMachine " , g_simulator . processesPerMachine ) ;
2017-05-26 04:48:44 +08:00
// SOMEDAY: add locality for testers to simulate network topology
// FIXME: Start workers with tester class instead, at least sometimes run tests with the testers-only flag
2019-05-11 05:01:52 +08:00
int testerCount = * pTesterCount = deterministicRandom ( ) - > randomInt ( 4 , 9 ) ;
int useSeedForMachine = deterministicRandom ( ) - > randomInt ( 0 , testerCount ) ;
2017-05-26 04:48:44 +08:00
for ( int i = 0 ; i < testerCount ; i + + ) {
2019-02-28 16:09:53 +08:00
std : : vector < IPAddress > ips ;
ips . push_back ( makeIPAddressForSim ( useIPv6 , { 3 , 4 , 3 , i + 1 } ) ) ;
2019-05-11 05:01:52 +08:00
Standalone < StringRef > newZoneId = Standalone < StringRef > ( deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ) ;
2017-05-26 04:48:44 +08:00
LocalityData localities ( Optional < Standalone < StringRef > > ( ) , newZoneId , newZoneId , Optional < Standalone < StringRef > > ( ) ) ;
systemActors - > push_back ( reportErrors ( simulatedMachine (
2020-02-11 06:00:21 +08:00
conn , ips , sslEnabled & & sslOnly ,
2017-05-26 04:48:44 +08:00
localities , ProcessClass ( ProcessClass : : TesterClass , ProcessClass : : CommandLineSource ) ,
2020-02-11 06:00:21 +08:00
baseFolder , false , i = = useSeedForMachine , AgentNone , sslEnabled & & sslOnly , whitelistBinPaths ) ,
2017-05-26 04:48:44 +08:00
" SimulatedTesterMachine " ) ) ;
}
* pStartingConfiguration = startingConfigString ;
// save some state that we only need when restarting the simulator.
g_simulator . connectionString = conn . toString ( ) ;
g_simulator . testerCount = testerCount ;
TraceEvent ( " SimulatedClusterStarted " )
. detail ( " DataCenters " , dataCenters )
. detail ( " ServerMachineCount " , machineCount )
. detail ( " ProcessesPerServer " , processesPerMachine )
. detail ( " SSLEnabled " , sslEnabled )
2019-02-01 10:20:14 +08:00
. detail ( " SSLOnly " , sslOnly )
2017-05-26 04:48:44 +08:00
. detail ( " ClassesAssigned " , assignClasses )
2018-02-27 05:15:44 +08:00
. detail ( " StartingConfiguration " , pStartingConfiguration - > toString ( ) ) ;
2017-05-26 04:48:44 +08:00
}
2020-02-12 03:10:09 +08:00
void checkTestConf ( const char * testFile , int & extraDB , int & minimumReplication , int & minimumRegions ,
2020-10-07 02:33:32 +08:00
int & configureLocked , int & logAntiQuorum ) {
2017-05-26 04:48:44 +08:00
std : : ifstream ifs ;
ifs . open ( testFile , std : : ifstream : : in ) ;
if ( ! ifs . good ( ) )
2017-09-16 08:55:01 +08:00
return ;
2017-05-26 04:48:44 +08:00
std : : string cline ;
while ( ifs . good ( ) ) {
getline ( ifs , cline ) ;
std : : string line = removeWhitespace ( std : : string ( cline ) ) ;
if ( ! line . size ( ) | | line . find ( ' ; ' ) = = 0 )
continue ;
size_t found = line . find ( ' = ' ) ;
if ( found = = std : : string : : npos )
// hmmm, not good
continue ;
std : : string attrib = removeWhitespace ( line . substr ( 0 , found ) ) ;
std : : string value = removeWhitespace ( line . substr ( found + 1 ) ) ;
if ( attrib = = " extraDB " ) {
2017-09-16 08:55:01 +08:00
sscanf ( value . c_str ( ) , " %d " , & extraDB ) ;
}
if ( attrib = = " minimumReplication " ) {
sscanf ( value . c_str ( ) , " %d " , & minimumReplication ) ;
2017-05-26 04:48:44 +08:00
}
2018-09-18 09:32:39 +08:00
if ( attrib = = " minimumRegions " ) {
sscanf ( value . c_str ( ) , " %d " , & minimumRegions ) ;
}
2020-02-12 03:10:09 +08:00
if ( attrib = = " configureLocked " ) {
sscanf ( value . c_str ( ) , " %d " , & configureLocked ) ;
}
2020-10-07 02:33:32 +08:00
if ( attrib = = " logAntiQuorum " ) {
sscanf ( value . c_str ( ) , " %d " , & logAntiQuorum ) ;
}
2017-05-26 04:48:44 +08:00
}
ifs . close ( ) ;
}
2020-02-07 11:29:31 +08:00
ACTOR void setupAndRun ( std : : string dataFolder , const char * testFile , bool rebooting , bool restoring , std : : string whitelistBinPaths ) {
2017-05-26 04:48:44 +08:00
state vector < Future < Void > > systemActors ;
state Optional < ClusterConnectionString > connFile ;
state Standalone < StringRef > startingConfiguration ;
state int testerCount = 1 ;
2017-09-16 08:55:01 +08:00
state int extraDB = 0 ;
state int minimumReplication = 0 ;
2018-09-18 09:32:39 +08:00
state int minimumRegions = 0 ;
2020-02-12 03:10:09 +08:00
state int configureLocked = 0 ;
2020-10-07 02:33:32 +08:00
state int logAntiQuorum = - 1 ;
checkTestConf ( testFile , extraDB , minimumReplication , minimumRegions , configureLocked , logAntiQuorum ) ;
2017-05-26 04:48:44 +08:00
2019-02-28 16:09:53 +08:00
// TODO (IPv6) Use IPv6?
wait ( g_simulator . onProcess (
2020-02-11 06:00:21 +08:00
g_simulator . newProcess ( " TestSystem " , IPAddress ( 0x01010101 ) , 1 , false , 1 ,
2019-02-28 16:09:53 +08:00
LocalityData ( Optional < Standalone < StringRef > > ( ) ,
2019-05-11 05:01:52 +08:00
Standalone < StringRef > ( deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ) ,
Standalone < StringRef > ( deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ) ,
2019-02-28 16:09:53 +08:00
Optional < Standalone < StringRef > > ( ) ) ,
ProcessClass ( ProcessClass : : TesterClass , ProcessClass : : CommandLineSource ) , " " , " " ) ,
2019-06-25 17:47:35 +08:00
TaskPriority : : DefaultYield ) ) ;
2017-05-26 04:48:44 +08:00
Sim2FileSystem : : newFileSystem ( ) ;
2019-04-19 05:12:45 +08:00
FlowTransport : : createInstance ( true , 1 ) ;
2017-05-26 04:48:44 +08:00
TEST ( true ) ; // Simulation start
try {
//systemActors.push_back( startSystemMonitor(dataFolder) );
if ( rebooting ) {
2020-02-07 11:29:31 +08:00
wait ( timeoutError ( restartSimulatedSystem ( & systemActors , dataFolder , & testerCount , & connFile , & startingConfiguration , extraDB , whitelistBinPaths ) , 100.0 ) ) ;
2019-04-03 20:47:19 +08:00
// FIXME: snapshot restore does not support multi-region restore, hence restore it as single region always
2019-03-26 09:31:08 +08:00
if ( restoring ) {
2019-04-13 04:23:02 +08:00
startingConfiguration = LiteralStringRef ( " usable_regions=1 " ) ;
2019-03-26 09:31:08 +08:00
}
}
else {
2017-05-26 04:48:44 +08:00
g_expect_full_pointermap = 1 ;
2019-02-28 16:09:53 +08:00
setupSimulatedSystem ( & systemActors , dataFolder , & testerCount , & connFile , & startingConfiguration , extraDB ,
2020-10-07 02:33:32 +08:00
minimumReplication , minimumRegions , whitelistBinPaths , configureLocked , logAntiQuorum ) ;
2018-08-11 04:57:10 +08:00
wait ( delay ( 1.0 ) ) ; // FIXME: WHY!!! //wait for machines to boot
2017-05-26 04:48:44 +08:00
}
2019-05-11 05:01:52 +08:00
std : : string clusterFileDir = joinPath ( dataFolder , deterministicRandom ( ) - > randomUniqueID ( ) . toString ( ) ) ;
2017-05-26 04:48:44 +08:00
platform : : createDirectory ( clusterFileDir ) ;
writeFile ( joinPath ( clusterFileDir , " fdb.cluster " ) , connFile . get ( ) . toString ( ) ) ;
2020-11-07 15:50:55 +08:00
wait ( timeoutError ( runTests ( makeReference < ClusterConnectionFile > ( joinPath ( clusterFileDir , " fdb.cluster " ) ) ,
TEST_TYPE_FROM_FILE , TEST_ON_TESTERS , testerCount , testFile , startingConfiguration ) ,
isBuggifyEnabled ( BuggifyType : : General ) ? 36000.0 : 5400.0 ) ) ;
2017-05-26 04:48:44 +08:00
} catch ( Error & e ) {
2018-06-09 04:57:00 +08:00
TraceEvent ( SevError , " SetupAndRunError " ) . error ( e ) ;
2017-05-26 04:48:44 +08:00
}
TraceEvent ( " SimulatedSystemDestruct " ) ;
2019-06-26 07:25:42 +08:00
g_simulator . stop ( ) ;
2017-05-26 04:48:44 +08:00
destructed = true ;
2019-07-10 05:01:05 +08:00
wait ( Never ( ) ) ;
ASSERT ( false ) ;
2017-05-26 04:48:44 +08:00
}