ClickHouse/dbms/src/Interpreters/ProcessList.h

302 lines
8.5 KiB
C++
Raw Normal View History

#pragma once
#include <map>
#include <list>
#include <memory>
#include <mutex>
#include <unordered_map>
#include <shared_mutex>
#include <Poco/Condition.h>
#include <Core/Defines.h>
#include <IO/Progress.h>
#include <Common/Stopwatch.h>
#include <Common/MemoryTracker.h>
#include <Common/CurrentMetrics.h>
#include <Common/ProfileEvents.h>
#include <Common/Throttler.h>
#include <Interpreters/QueryPriorities.h>
#include <Interpreters/ClientInfo.h>
#include <Common/ThreadStatus.h>
#include <DataStreams/BlockIO.h>
#include "ThreadPerformanceProfile.h"
namespace CurrentMetrics
{
extern const Metric Query;
}
namespace DB
{
class IStorage;
using StoragePtr = std::shared_ptr<IStorage>;
using Tables = std::map<String, StoragePtr>;
struct Settings;
2017-01-27 17:27:33 +00:00
class IAST;
struct ProcessListForUser;
struct QueryStatus;
struct ThreadStatus;
/** List of currently executing queries.
* Also implements limit on their number.
*/
/** Information of process list element.
* To output in SHOW PROCESSLIST query. Does not contain any complex objects, that do something on copy or destructor.
*/
struct QueryStatusInfo
{
String query;
double elapsed_seconds;
size_t read_rows;
size_t read_bytes;
size_t total_rows;
size_t written_rows;
size_t written_bytes;
Int64 memory_usage;
ClientInfo client_info;
};
/// Query and information about its execution.
struct QueryStatus
{
String query;
ClientInfo client_info;
2014-09-10 11:34:26 +00:00
Stopwatch watch;
/// Progress of input stream
Progress progress_in;
/// Progress of output stream
Progress progress_out;
QueryPriorities::Handle priority_handle;
ProfileEvents::Counters performance_counters;
MemoryTracker memory_tracker;
mutable std::shared_mutex threads_mutex;
using QueryThreadStatuses = std::map<int, ThreadStatusPtr>; /// Key is Poco's thread_id
QueryThreadStatuses thread_statuses;
CurrentMetrics::Increment num_queries_increment{CurrentMetrics::Query};
bool is_cancelled = false;
2014-09-10 11:34:26 +00:00
/// Temporary tables could be registered here. Modify under mutex.
Tables temporary_tables;
void setUserProcessList(ProcessListForUser * user_process_list_);
/// Be careful using it. For example, queries field of ProcessListForUser could be modified concurrently.
const ProcessListForUser * getUserProcessList() const { return user_process_list; }
2016-11-30 17:31:05 +00:00
protected:
mutable std::mutex query_streams_mutex;
/// Streams with query results, point to BlockIO from executeQuery()
/// This declaration is compatible with notes about BlockIO::process_list_entry:
/// there are no cyclic dependencies: BlockIO::in,out point to objects inside ProcessListElement (not whole object)
BlockInputStreamPtr query_stream_in;
BlockOutputStreamPtr query_stream_out;
2016-11-30 17:31:05 +00:00
bool query_streams_initialized{false};
bool query_streams_released{false};
2016-11-30 17:31:05 +00:00
ProcessListForUser * user_process_list = nullptr;
2016-11-30 17:31:05 +00:00
public:
QueryStatus(
const String & query_,
const ClientInfo & client_info_,
size_t max_memory_usage,
double memory_tracker_fault_probability,
QueryPriorities::Handle && priority_handle_);
~QueryStatus()
{
// TODO: master thread should be reset
}
bool updateProgressIn(const Progress & value)
{
progress_in.incrementPiecewiseAtomically(value);
if (priority_handle)
priority_handle->waitIfNeed(std::chrono::seconds(1)); /// NOTE Could make timeout customizable.
return !is_cancelled;
}
bool updateProgressOut(const Progress & value)
{
progress_out.incrementPiecewiseAtomically(value);
return !is_cancelled;
}
QueryStatusInfo getInfo() const
{
QueryStatusInfo res;
res.query = query;
2017-04-23 08:29:24 +00:00
res.client_info = client_info;
res.elapsed_seconds = watch.elapsedSeconds();
res.read_rows = progress_in.rows;
res.read_bytes = progress_in.bytes;
res.total_rows = progress_in.total_rows;
2017-04-23 08:29:24 +00:00
res.written_rows = progress_out.rows;
res.written_bytes = progress_out.bytes;
res.memory_usage = memory_tracker.get();
return res;
}
/// Copies pointers to in/out streams
void setQueryStreams(const BlockIO & io);
/// Frees in/out streams
void releaseQueryStreams();
/// It means that ProcessListEntry still exists, but stream was already destroyed
bool streamsAreReleased();
/// Get query in/out pointers from BlockIO
bool tryGetQueryStreams(BlockInputStreamPtr & in, BlockOutputStreamPtr & out) const;
2015-04-16 06:12:35 +00:00
};
/// Data about queries for one user.
struct ProcessListForUser
{
ProcessListForUser();
/// Query_id -> ProcessListElement *
using QueryToElement = std::unordered_map<String, QueryStatus *>;
QueryToElement queries;
ProfileEvents::Counters user_performance_counters;
/// Limit and counter for memory of all simultaneously running queries of single user.
MemoryTracker user_memory_tracker;
2017-08-29 20:20:21 +00:00
/// Count network usage for all simultaneously running queries of single user.
ThrottlerPtr user_throttler;
};
class ProcessList;
/// Keeps iterator to process list and removes element in destructor.
class ProcessListEntry
{
private:
using Container = std::list<QueryStatus>;
ProcessList & parent;
Container::iterator it;
public:
ProcessListEntry(ProcessList & parent_, Container::iterator it_)
: parent(parent_), it(it_) {}
~ProcessListEntry();
QueryStatus * operator->() { return &*it; }
const QueryStatus * operator->() const { return &*it; }
QueryStatus & get() { return *it; }
const QueryStatus & get() const { return *it; }
};
2015-04-16 06:12:35 +00:00
class ProcessList
{
friend class ProcessListEntry;
2015-04-16 06:12:35 +00:00
public:
using Element = QueryStatus;
using Entry = ProcessListEntry;
/// list, for iterators not to invalidate. NOTE: could replace with cyclic buffer, but not worth.
using Container = std::list<Element>;
using Info = std::vector<QueryStatusInfo>;
/// User -> queries
using UserToQueries = std::unordered_map<String, ProcessListForUser>;
private:
mutable std::mutex mutex;
mutable Poco::Condition have_space; /// Number of currently running queries has become less than maximum.
2014-09-10 11:34:26 +00:00
/// List of queries
Container processes;
size_t max_size; /// 0 means no limit. Otherwise, when limit exceeded, an exception is thrown.
/// Stores per-user info: queries, statistics and limits
UserToQueries user_to_queries;
/// Stores info about queries grouped by their priority
QueryPriorities priorities;
/// Limit and counter for memory of all simultaneously running queries.
MemoryTracker total_memory_tracker;
/// Call under lock. Finds process with specified current_user and current_query_id.
QueryStatus * tryGetProcessListElement(const String & current_query_id, const String & current_user);
public:
ProcessList(size_t max_size_ = 0) : max_size(max_size_) {}
using EntryPtr = std::shared_ptr<ProcessListEntry>;
/** Register running query. Returns refcounted object, that will remove element from list in destructor.
2017-11-15 19:47:49 +00:00
* If too many running queries - wait for not more than specified (see settings) amount of time.
* If timeout is passed - throw an exception.
* Don't count KILL QUERY queries.
*/
EntryPtr insert(const String & query_, const IAST * ast, const ClientInfo & client_info, const Settings & settings);
/// Number of currently executing queries.
size_t size() const { return processes.size(); }
/// Get current state of process list.
Info getInfo() const
{
std::lock_guard<std::mutex> lock(mutex);
Info res;
res.reserve(processes.size());
for (const auto & elem : processes)
res.emplace_back(elem.getInfo());
return res;
}
void setMaxSize(size_t max_size_)
{
std::lock_guard<std::mutex> lock(mutex);
max_size = max_size_;
}
/// Register temporary table. Then it is accessible by query_id and name.
void addTemporaryTable(QueryStatus & elem, const String & table_name, const StoragePtr & storage);
enum class CancellationCode
{
NotFound = 0, /// already cancelled
QueryIsNotInitializedYet = 1,
CancelCannotBeSent = 2,
CancelSent = 3,
Unknown
};
2016-11-30 17:31:05 +00:00
/// Try call cancel() for input and output streams of query with specified id and user
CancellationCode sendCancelToQuery(const String & current_query_id, const String & current_user, bool kill = false);
};
}