From 99188db967a599d179cf65a80d303244ed3f1e81 Mon Sep 17 00:00:00 2001 From: Andrew Chang Date: Wed, 8 Jan 2025 15:40:06 -0800 Subject: [PATCH 1/3] Add test_follower flag to crash test --- db_stress_tool/db_stress_common.h | 2 ++ db_stress_tool/db_stress_gflags.cc | 7 +++++++ db_stress_tool/db_stress_stat.cc | 1 + db_stress_tool/db_stress_stat.h | 1 + db_stress_tool/db_stress_test_base.cc | 4 ++++ db_stress_tool/db_stress_tool.cc | 16 ++++++++++++++++ db_stress_tool/multi_ops_txns_stress.cc | 10 +++++----- 7 files changed, 36 insertions(+), 5 deletions(-) diff --git a/db_stress_tool/db_stress_common.h b/db_stress_tool/db_stress_common.h index 0c1ef4e1136..d51dd3ab63d 100644 --- a/db_stress_tool/db_stress_common.h +++ b/db_stress_tool/db_stress_common.h @@ -176,7 +176,9 @@ DECLARE_int32(index_type); DECLARE_int32(data_block_index_type); DECLARE_string(db); DECLARE_string(secondaries_base); +DECLARE_string(followers_base); DECLARE_bool(test_secondary); +DECLARE_bool(test_follower); DECLARE_string(expected_values_dir); DECLARE_bool(verify_checksum); DECLARE_bool(mmap_read); diff --git a/db_stress_tool/db_stress_gflags.cc b/db_stress_tool/db_stress_gflags.cc index 1859e6940fb..42147fb1e10 100644 --- a/db_stress_tool/db_stress_gflags.cc +++ b/db_stress_tool/db_stress_gflags.cc @@ -618,10 +618,17 @@ DEFINE_string(db, "", "Use the db with the following name."); DEFINE_string(secondaries_base, "", "Use this path as the base path for secondary instances."); +DEFINE_string(followers_base, "", + "Use this path as the base path for follower instances."); + DEFINE_bool(test_secondary, false, "If true, start an additional secondary instance which can be used " "for verification."); +DEFINE_bool(test_follower, false, + "If true, start an additional follower instance which can be used " + "for verification."); + DEFINE_string( expected_values_dir, "", "Dir where files containing info about the latest/historical values will " diff --git a/db_stress_tool/db_stress_stat.cc b/db_stress_tool/db_stress_stat.cc index 6a7883a52ac..c2e8d127e03 100644 --- a/db_stress_tool/db_stress_stat.cc +++ b/db_stress_tool/db_stress_stat.cc @@ -11,6 +11,7 @@ namespace ROCKSDB_NAMESPACE { std::shared_ptr dbstats; std::shared_ptr dbstats_secondaries; +std::shared_ptr dbstats_followers; } // namespace ROCKSDB_NAMESPACE diff --git a/db_stress_tool/db_stress_stat.h b/db_stress_tool/db_stress_stat.h index 5b38c6e2bb5..0eb2463f6fd 100644 --- a/db_stress_tool/db_stress_stat.h +++ b/db_stress_tool/db_stress_stat.h @@ -25,6 +25,7 @@ namespace ROCKSDB_NAMESPACE { // Database statistics extern std::shared_ptr dbstats; extern std::shared_ptr dbstats_secondaries; +extern std::shared_ptr dbstats_followers; class Stats { private: diff --git a/db_stress_tool/db_stress_test_base.cc b/db_stress_tool/db_stress_test_base.cc index 3200658fe8c..774ff631192 100644 --- a/db_stress_tool/db_stress_test_base.cc +++ b/db_stress_tool/db_stress_test_base.cc @@ -645,6 +645,10 @@ void StressTest::PrintStatistics() { fprintf(stdout, "Secondary instances STATISTICS:\n%s\n", dbstats_secondaries->ToString().c_str()); } + if (dbstats_followers) { + fprintf(stdout, "Follower instances STATISTICS:\n%s\n", + dbstats_followers->ToString().c_str()); + } } // Currently PreloadDb has to be single-threaded. diff --git a/db_stress_tool/db_stress_tool.cc b/db_stress_tool/db_stress_tool.cc index ca43b699c8f..55b8530b7f6 100644 --- a/db_stress_tool/db_stress_tool.cc +++ b/db_stress_tool/db_stress_tool.cc @@ -58,6 +58,9 @@ int db_stress_tool(int argc, char** argv) { if (FLAGS_test_secondary) { dbstats_secondaries = ROCKSDB_NAMESPACE::CreateDBStatistics(); } + if (FLAGS_test_follower) { + dbstats_followers = ROCKSDB_NAMESPACE::CreateDBStatistics(); + } } compression_type_e = StringToCompressionType(FLAGS_compression_type.c_str()); bottommost_compression_type_e = @@ -231,6 +234,19 @@ int db_stress_tool(int argc, char** argv) { FLAGS_secondaries_base = default_secondaries_path; } + if (FLAGS_test_follower && FLAGS_followers_base.empty()) { + std::string default_followers_path; + db_stress_env->GetTestDirectory(&default_followers_path); + default_followers_path += "/dbstress_followers"; + s = db_stress_env->CreateDirIfMissing(default_followers_path); + if (!s.ok()) { + fprintf(stderr, "Failed to create directory %s: %s\n", + default_followers_path.c_str(), s.ToString().c_str()); + exit(1); + } + FLAGS_followers_base = default_followers_path; + } + if (FLAGS_best_efforts_recovery && !(FLAGS_skip_verifydb && FLAGS_disable_wal)) { fprintf(stderr, diff --git a/db_stress_tool/multi_ops_txns_stress.cc b/db_stress_tool/multi_ops_txns_stress.cc index 6ec42e4e7fb..7538eea64f5 100644 --- a/db_stress_tool/multi_ops_txns_stress.cc +++ b/db_stress_tool/multi_ops_txns_stress.cc @@ -1759,11 +1759,11 @@ void CheckAndSetOptionsForMultiOpsTxnStressTest() { if (!FLAGS_use_txn) { fprintf(stderr, "-use_txn must be true if -test_multi_ops_txns\n"); exit(1); - } else if (FLAGS_test_secondary > 0) { - fprintf( - stderr, - "secondary instance does not support replaying logs (MANIFEST + WAL) " - "of TransactionDB with write-prepared/write-unprepared policy\n"); + } else if (FLAGS_test_secondary > 0 || FLAGS_test_follower > 0) { + fprintf(stderr, + "secondary and follower instances do not support replaying logs " + "(MANIFEST + WAL) " + "of TransactionDB with write-prepared/write-unprepared policy\n"); exit(1); } if (FLAGS_clear_column_family_one_in > 0) { From d75b27e41f0afa6927db0e0b46567b30d73a4a46 Mon Sep 17 00:00:00 2001 From: Andrew Chang Date: Thu, 9 Jan 2025 12:28:14 -0800 Subject: [PATCH 2/3] Add OpenAsFollower call --- db_stress_tool/db_stress_test_base.cc | 19 ++++++++++++++++--- db_stress_tool/db_stress_test_base.h | 2 ++ 2 files changed, 18 insertions(+), 3 deletions(-) diff --git a/db_stress_tool/db_stress_test_base.cc b/db_stress_tool/db_stress_test_base.cc index 774ff631192..f25b36599ff 100644 --- a/db_stress_tool/db_stress_test_base.cc +++ b/db_stress_tool/db_stress_test_base.cc @@ -3690,9 +3690,9 @@ void StressTest::Open(SharedState* shared, bool reopen) { assert(column_families_.size() == static_cast(FLAGS_column_families)); - // Secondary instance does not support write-prepared/write-unprepared - // transactions, thus just disable secondary instance if we use - // transaction. + // Secondary and follower instances do not support + // write-prepared/write-unprepared transactions, thus just disable secondary + // instance if we use transaction. if (s.ok() && FLAGS_test_secondary && !FLAGS_use_txn) { Options tmp_opts; // TODO(yanqin) support max_open_files != -1 for secondary instance. @@ -3705,6 +3705,19 @@ void StressTest::Open(SharedState* shared, bool reopen) { assert(secondary_cfhs_.size() == static_cast(FLAGS_column_families)); } + + if (s.ok() && FLAGS_test_follower && !FLAGS_use_txn) { + Options tmp_opts; + tmp_opts.max_open_files = -1; + tmp_opts.env = db_stress_env; + // Need to use the leader path not the follower path + const std::string& leader_path = FLAGS_db; + s = DB::OpenAsFollower(tmp_opts, FLAGS_db /* name */, leader_path, + cf_descriptors, &follower_cfhs_, &follower_db_); + assert(s.ok()); + assert(follower_cfhs_.size() == + static_cast(FLAGS_column_families)); + } } else { DBWithTTL* db_with_ttl; s = DBWithTTL::Open(options_, FLAGS_db, &db_with_ttl, FLAGS_ttl); diff --git a/db_stress_tool/db_stress_test_base.h b/db_stress_tool/db_stress_test_base.h index 25e4cdedf27..185249e4919 100644 --- a/db_stress_tool/db_stress_test_base.h +++ b/db_stress_tool/db_stress_test_base.h @@ -414,6 +414,8 @@ class StressTest { DB* secondary_db_; std::vector secondary_cfhs_; + std::unique_ptr follower_db_; + std::vector follower_cfhs_; bool is_db_stopped_; }; From 9b2794cdc1103eaca67ca53b1331dfc7c2602c7e Mon Sep 17 00:00:00 2001 From: Andrew Chang Date: Thu, 9 Jan 2025 14:48:19 -0800 Subject: [PATCH 3/3] Use followers_base as name in OpenAsFollower --- db_stress_tool/db_stress_test_base.cc | 11 ++++++++--- tools/db_crashtest.py | 1 + 2 files changed, 9 insertions(+), 3 deletions(-) diff --git a/db_stress_tool/db_stress_test_base.cc b/db_stress_tool/db_stress_test_base.cc index f25b36599ff..a29f165896b 100644 --- a/db_stress_tool/db_stress_test_base.cc +++ b/db_stress_tool/db_stress_test_base.cc @@ -3710,10 +3710,15 @@ void StressTest::Open(SharedState* shared, bool reopen) { Options tmp_opts; tmp_opts.max_open_files = -1; tmp_opts.env = db_stress_env; - // Need to use the leader path not the follower path + // Equivalent to "name" argument in OpenAsSecondary const std::string& leader_path = FLAGS_db; - s = DB::OpenAsFollower(tmp_opts, FLAGS_db /* name */, leader_path, - cf_descriptors, &follower_cfhs_, &follower_db_); + // Equivalent to "secondary_path" in OpenAsSecondary + const std::string& name = FLAGS_followers_base; + s = DB::OpenAsFollower(tmp_opts, name, leader_path, cf_descriptors, + &follower_cfhs_, &follower_db_); + if (!s.ok()) { + fprintf(stderr, "Error opening follower: %s\n", s.ToString().c_str()); + } assert(s.ok()); assert(follower_cfhs_.size() == static_cast(FLAGS_column_families)); diff --git a/tools/db_crashtest.py b/tools/db_crashtest.py index 615b1e448f4..f6a5f1c5e1a 100644 --- a/tools/db_crashtest.py +++ b/tools/db_crashtest.py @@ -486,6 +486,7 @@ def is_direct_io_supported(dbname): "level_compaction_dynamic_level_bytes": lambda: random.randint(0, 1), "paranoid_file_checks": lambda: random.choice([0, 1, 1, 1]), "test_secondary": lambda: random.choice([0, 1]), + "test_follower": lambda: random.choice([0, 1]), } blackbox_simple_default_params = {