diff --git a/velox/common/base/PeriodicStatsReporter.cpp b/velox/common/base/PeriodicStatsReporter.cpp index 04fbaa104012..f3c3b6e5b608 100644 --- a/velox/common/base/PeriodicStatsReporter.cpp +++ b/velox/common/base/PeriodicStatsReporter.cpp @@ -26,12 +26,38 @@ namespace { if ((counter) != 0) { \ RECORD_METRIC_VALUE((name), (counter)); \ } + +std::mutex& instanceMutex() { + static std::mutex instanceMu; + return instanceMu; +} + +// Global instance. Must be called while holding a lock over instanceMutex(). +std::unique_ptr& instance() { + static std::unique_ptr reporter; + return reporter; +} } // namespace -PeriodicStatsReporter::PeriodicStatsReporter( - const velox::memory::MemoryArbitrator* arbitrator, - const Options& options) - : arbitrator_(arbitrator), options_(options) {} +void startPeriodicStatsReporter(const PeriodicStatsReporter::Options& options) { + std::lock_guard l(instanceMutex()); + auto& instanceRef = instance(); + VELOX_CHECK_NULL( + instanceRef, "The periodic stats reporter has already started."); + instanceRef = std::make_unique(options); + instanceRef->start(); +} + +void stopPeriodicStatsReporter() { + std::lock_guard l(instanceMutex()); + auto& instanceRef = instance(); + VELOX_CHECK_NOT_NULL(instanceRef, "No periodic stats reporter to stop."); + instanceRef->stop(); + instanceRef.reset(); +} + +PeriodicStatsReporter::PeriodicStatsReporter(const Options& options) + : arbitrator_(options.arbitrator), options_(options) {} void PeriodicStatsReporter::start() { LOG(INFO) << "Starting PeriodicStatsReporter with options " diff --git a/velox/common/base/PeriodicStatsReporter.h b/velox/common/base/PeriodicStatsReporter.h index b73332605ee6..7621ac99a01c 100644 --- a/velox/common/base/PeriodicStatsReporter.h +++ b/velox/common/base/PeriodicStatsReporter.h @@ -39,6 +39,8 @@ class PeriodicStatsReporter { struct Options { Options() {} + const memory::MemoryArbitrator* arbitrator{nullptr}; + uint64_t arbitratorStatsIntervalMs{60'000}; std::string toString() const { @@ -47,9 +49,7 @@ class PeriodicStatsReporter { } }; - PeriodicStatsReporter( - const velox::memory::MemoryArbitrator* arbitrator, - const Options& options = Options()); + PeriodicStatsReporter(const Options& options = Options()); /// Invoked to start the report daemon in background. void start(); @@ -83,4 +83,13 @@ class PeriodicStatsReporter { folly::ThreadedRepeatingFunctionRunner scheduler_; }; + +/// Initializes and starts the process-wide periodic stats reporter. Before +/// 'stopPeriodicStatsReporter()' is called, this method can only be called once +/// process-wide, and additional calls to this method will throw. +void startPeriodicStatsReporter(const PeriodicStatsReporter::Options& options); + +/// Stops the process-wide periodic stats reporter. +void stopPeriodicStatsReporter(); + } // namespace facebook::velox diff --git a/velox/common/base/tests/StatsReporterTest.cpp b/velox/common/base/tests/StatsReporterTest.cpp index 241a68c6af61..fd7e89a6f35f 100644 --- a/velox/common/base/tests/StatsReporterTest.cpp +++ b/velox/common/base/tests/StatsReporterTest.cpp @@ -23,6 +23,7 @@ #include #include "velox/common/base/Counters.h" #include "velox/common/base/PeriodicStatsReporter.h" +#include "velox/common/base/tests/GTestUtils.h" namespace facebook::velox { @@ -196,8 +197,9 @@ class TestStatsReportMemoryArbitrator : public memory::MemoryArbitrator { TEST_F(PeriodicStatsReporterTest, basic) { TestStatsReportMemoryArbitrator arbitrator({}); PeriodicStatsReporter::Options options; + options.arbitrator = &arbitrator; options.arbitratorStatsIntervalMs = 4'000; - PeriodicStatsReporter periodicReporter(&arbitrator, options); + PeriodicStatsReporter periodicReporter(options); periodicReporter.start(); std::this_thread::sleep_for(std::chrono::milliseconds(2'000)); @@ -212,6 +214,20 @@ TEST_F(PeriodicStatsReporterTest, basic) { counterMap.count(kMetricArbitratorFreeReservedCapacityBytes.str()), 1); } +TEST_F(PeriodicStatsReporterTest, globalInstance) { + TestStatsReportMemoryArbitrator arbitrator({}); + PeriodicStatsReporter::Options options; + options.arbitrator = &arbitrator; + options.arbitratorStatsIntervalMs = 4'000; + VELOX_ASSERT_THROW( + stopPeriodicStatsReporter(), "No periodic stats reporter to stop."); + ASSERT_NO_THROW(startPeriodicStatsReporter(options)); + VELOX_ASSERT_THROW( + startPeriodicStatsReporter(options), + "The periodic stats reporter has already started."); + ASSERT_NO_THROW(stopPeriodicStatsReporter()); +} + // Registering to folly Singleton with intended reporter type folly::Singleton reporter([]() { return new TestReporter();