#include #include #include #include #include namespace ProfileEvents { extern const Event ThrottlerSleepMicroseconds; } namespace DB { namespace ErrorCodes { extern const int LIMIT_EXCEEDED; } /// Just 10^9. static constexpr auto NS = 1000000000UL; Throttler::Throttler(size_t max_speed_, const std::shared_ptr & parent_) : max_speed(max_speed_) , max_burst(max_speed_ * default_burst_seconds) , limit_exceeded_exception_message("") , tokens(max_burst) , parent(parent_) {} Throttler::Throttler(size_t max_speed_, size_t limit_, const char * limit_exceeded_exception_message_, const std::shared_ptr & parent_) : max_speed(max_speed_) , max_burst(max_speed_ * default_burst_seconds) , limit(limit_) , limit_exceeded_exception_message(limit_exceeded_exception_message_) , tokens(max_burst) , parent(parent_) {} UInt64 Throttler::add(size_t amount) { // Values obtained under lock to be checked after release size_t count_value; double tokens_value; { std::lock_guard lock(mutex); auto now = clock_gettime_ns_adjusted(prev_ns); if (max_speed) { double delta_seconds = prev_ns ? static_cast(now - prev_ns) / NS : 0; tokens = std::min(tokens + max_speed * delta_seconds - amount, max_burst); } count += amount; count_value = count; tokens_value = tokens; prev_ns = now; } if (limit && count_value > limit) throw Exception(limit_exceeded_exception_message + std::string(" Maximum: ") + toString(limit), ErrorCodes::LIMIT_EXCEEDED); /// Wait unless there is positive amount of tokens - throttling Int64 sleep_time = 0; if (max_speed && tokens_value < 0) { sleep_time = static_cast(-tokens_value / max_speed * NS); accumulated_sleep += sleep_time; sleepForNanoseconds(sleep_time); accumulated_sleep -= sleep_time; ProfileEvents::increment(ProfileEvents::ThrottlerSleepMicroseconds, sleep_time / 1000UL); } if (parent) sleep_time += parent->add(amount); return static_cast(sleep_time); } void Throttler::reset() { std::lock_guard lock(mutex); count = 0; tokens = max_burst; prev_ns = 0; // NOTE: do not zero `accumulated_sleep` to avoid races } bool Throttler::isThrottling() const { if (accumulated_sleep != 0) return true; if (parent) return parent->isThrottling(); return false; } }