#include #include #include #include #include namespace DB { Block MetricLogElement::createBlock() { ColumnsWithTypeAndName columns_with_type_and_name; columns_with_type_and_name.emplace_back(std::make_shared(), "event_date"); columns_with_type_and_name.emplace_back(std::make_shared(), "event_time"); columns_with_type_and_name.emplace_back(std::make_shared(), "milliseconds"); //ProfileEvents for (size_t i = 0, end = ProfileEvents::end(); i < end; ++i) { std::string name; name += "ProfileEvent_"; name += ProfileEvents::getName(ProfileEvents::Event(i)); columns_with_type_and_name.emplace_back(std::make_shared(), std::move(name)); } //CurrentMetrics for (size_t i = 0, end = CurrentMetrics::end(); i < end; ++i) { std::string name; name += "CurrentMetric_"; name += CurrentMetrics::getName(ProfileEvents::Event(i)); columns_with_type_and_name.emplace_back(std::make_shared(), std::move(name)); } return Block(columns_with_type_and_name); } void MetricLogElement::appendToBlock(Block & block) const { MutableColumns columns = block.mutateColumns(); size_t iter = 0; columns[iter++]->insert(DateLUT::instance().toDayNum(event_time)); columns[iter++]->insert(event_time); columns[iter++]->insert(milliseconds); //ProfileEvents for (size_t i = 0, end = ProfileEvents::end(); i < end; ++i) { const UInt64 value = ProfileEvents::global_counters[i].load(std::memory_order_relaxed); columns[iter++]->insert(value); } //CurrentMetrics for (size_t i = 0, end = CurrentMetrics::end(); i < end; ++i) { const UInt64 value = CurrentMetrics::values[i]; columns[iter++]->insert(value); } } void MetricLog::startCollectMetric(size_t collect_interval_milliseconds_) { collect_interval_milliseconds = collect_interval_milliseconds_; is_shutdown_metric_thread = false; metric_flush_thread = ThreadFromGlobalPool([this] { metricThreadFunction(); }); } void MetricLog::stopCollectMetric() { bool old_val = false; if (!is_shutdown_metric_thread.compare_exchange_strong(old_val, true)) return; metric_flush_thread.join(); } inline UInt64 time_in_milliseconds(std::chrono::time_point timepoint) { return std::chrono::duration_cast(timepoint.time_since_epoch()).count(); } inline UInt64 time_in_seconds(std::chrono::time_point timepoint) { return std::chrono::duration_cast(timepoint.time_since_epoch()).count(); } void MetricLog::metricThreadFunction() { auto desired_timepoint = std::chrono::system_clock::now(); while (!is_shutdown_metric_thread) { try { MetricLogElement elem; const auto current_time = std::chrono::system_clock::now(); elem.event_time = std::chrono::system_clock::to_time_t(current_time); elem.milliseconds = time_in_milliseconds(current_time) - time_in_seconds(current_time) * 1000; this->add(elem); /// We will record current time into table but align it to regular time intervals to avoid time drift. /// We may drop some time points if the server is overloaded and recording took too much time. while (desired_timepoint <= current_time) desired_timepoint += std::chrono::milliseconds(collect_interval_milliseconds); std::this_thread::sleep_until(desired_timepoint); } catch (...) { tryLogCurrentException(__PRETTY_FUNCTION__); } } } }