2011-08-27 22:43:31 +00:00
|
|
|
#include <iomanip>
|
2016-05-23 00:40:28 +00:00
|
|
|
#include <random>
|
2011-08-27 22:43:31 +00:00
|
|
|
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Interpreters/Quota.h>
|
|
|
|
#include <Interpreters/ProcessList.h>
|
|
|
|
#include <DataStreams/IProfilingBlockInputStream.h>
|
2018-02-19 03:56:08 +00:00
|
|
|
#include <Columns/ColumnConst.h>
|
|
|
|
|
2011-08-27 22:43:31 +00:00
|
|
|
|
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
|
2016-01-11 21:46:36 +00:00
|
|
|
namespace ErrorCodes
|
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
extern const int TOO_MUCH_ROWS;
|
|
|
|
extern const int TOO_MUCH_BYTES;
|
|
|
|
extern const int TIMEOUT_EXCEEDED;
|
|
|
|
extern const int TOO_SLOW;
|
2017-07-27 21:27:10 +00:00
|
|
|
extern const int LOGICAL_ERROR;
|
2018-02-19 03:56:08 +00:00
|
|
|
extern const int BLOCKS_HAVE_DIFFERENT_STRUCTURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-02-19 19:18:51 +00:00
|
|
|
void IProfilingBlockInputStream::checkBlockStructure(const Block & block, const Block & header)
|
2018-02-19 03:56:08 +00:00
|
|
|
{
|
|
|
|
size_t columns = header.columns();
|
|
|
|
if (block.columns() != columns)
|
2018-02-19 19:18:51 +00:00
|
|
|
throw Exception("Block structure mismatch in " + getName() + " stream: different number of columns:\n"
|
2018-02-19 03:56:08 +00:00
|
|
|
+ block.dumpStructure() + "\n" + header.dumpStructure(), ErrorCodes::BLOCKS_HAVE_DIFFERENT_STRUCTURE);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < columns; ++i)
|
|
|
|
{
|
|
|
|
const auto & expected = header.getByPosition(i);
|
|
|
|
const auto & actual = block.getByPosition(i);
|
|
|
|
|
|
|
|
if (actual.name != expected.name)
|
2018-02-19 19:18:51 +00:00
|
|
|
throw Exception("Block structure mismatch in " + getName() + " stream: different names of columns:\n"
|
|
|
|
+ block.dumpStructure() + "\n" + header.dumpStructure(), ErrorCodes::BLOCKS_HAVE_DIFFERENT_STRUCTURE);
|
2018-02-19 03:56:08 +00:00
|
|
|
|
|
|
|
if (!actual.type->equals(*expected.type))
|
2018-02-19 19:18:51 +00:00
|
|
|
throw Exception("Block structure mismatch in " + getName() + " stream: different types:\n"
|
|
|
|
+ block.dumpStructure() + "\n" + header.dumpStructure(), ErrorCodes::BLOCKS_HAVE_DIFFERENT_STRUCTURE);
|
2018-02-19 03:56:08 +00:00
|
|
|
|
|
|
|
if (actual.column->getName() != expected.column->getName())
|
2018-02-19 19:18:51 +00:00
|
|
|
throw Exception("Block structure mismatch in " + getName() + " stream: different columns:\n"
|
|
|
|
+ block.dumpStructure() + "\n" + header.dumpStructure(), ErrorCodes::BLOCKS_HAVE_DIFFERENT_STRUCTURE);
|
2018-02-19 03:56:08 +00:00
|
|
|
|
|
|
|
if (actual.column->isColumnConst() && expected.column->isColumnConst()
|
|
|
|
&& static_cast<const ColumnConst &>(*actual.column).getField() != static_cast<const ColumnConst &>(*expected.column).getField())
|
2018-02-19 19:18:51 +00:00
|
|
|
throw Exception("Block structure mismatch in " + getName() + " stream: different values of constants",
|
|
|
|
ErrorCodes::BLOCKS_HAVE_DIFFERENT_STRUCTURE);
|
2018-02-19 03:56:08 +00:00
|
|
|
}
|
2016-01-11 21:46:36 +00:00
|
|
|
}
|
|
|
|
|
2011-08-27 22:43:31 +00:00
|
|
|
|
2017-06-26 12:30:35 +00:00
|
|
|
IProfilingBlockInputStream::IProfilingBlockInputStream()
|
|
|
|
{
|
|
|
|
info.parent = this;
|
|
|
|
}
|
|
|
|
|
2011-09-04 21:23:19 +00:00
|
|
|
Block IProfilingBlockInputStream::read()
|
2011-08-27 22:43:31 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
collectAndSendTotalRowsApprox();
|
2015-02-15 06:08:31 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
if (!info.started)
|
|
|
|
{
|
|
|
|
info.total_stopwatch.start();
|
|
|
|
info.started = true;
|
|
|
|
}
|
2012-05-09 08:16:09 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
Block res;
|
2013-11-17 19:14:17 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
if (is_cancelled.load(std::memory_order_seq_cst))
|
|
|
|
return res;
|
2012-05-31 00:47:05 +00:00
|
|
|
|
2018-01-12 13:30:57 +00:00
|
|
|
if (!checkTimeLimits())
|
2017-12-23 00:30:26 +00:00
|
|
|
limit_exceeded_need_break = true;
|
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
if (!limit_exceeded_need_break)
|
|
|
|
res = readImpl();
|
2011-08-27 22:43:31 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
if (res)
|
|
|
|
{
|
|
|
|
info.update(res);
|
2013-09-07 02:03:13 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
if (enabled_extremes)
|
|
|
|
updateExtremes(res);
|
2013-09-07 02:03:13 +00:00
|
|
|
|
2018-01-12 13:30:57 +00:00
|
|
|
if (!checkDataSizeLimits())
|
|
|
|
limit_exceeded_need_break = true;
|
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
if (quota != nullptr)
|
|
|
|
checkQuota(res);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/** If the thread is over, then we will ask all children to abort the execution.
|
|
|
|
* This makes sense when running a query with LIMIT
|
|
|
|
* - there is a situation when all the necessary data has already been read,
|
|
|
|
* but `children sources are still working,
|
|
|
|
* herewith they can work in separate threads or even remotely.
|
|
|
|
*/
|
|
|
|
cancel();
|
|
|
|
}
|
2011-09-25 03:37:09 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
progress(Progress(res.rows(), res.bytes()));
|
2012-12-25 17:25:44 +00:00
|
|
|
|
2018-02-19 03:56:08 +00:00
|
|
|
#ifndef NDEBUG
|
|
|
|
if (res)
|
|
|
|
{
|
|
|
|
Block header = getHeader();
|
|
|
|
if (header)
|
|
|
|
checkBlockStructure(res, header);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
return res;
|
2013-09-07 02:03:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-05-20 20:01:34 +00:00
|
|
|
void IProfilingBlockInputStream::readPrefix()
|
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
readPrefixImpl();
|
2016-05-31 01:04:34 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
for (auto & child : children)
|
|
|
|
child->readPrefix();
|
2016-05-20 20:01:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-13 20:33:09 +00:00
|
|
|
void IProfilingBlockInputStream::readSuffix()
|
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
for (auto & child : children)
|
|
|
|
child->readSuffix();
|
2013-09-13 20:33:09 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
readSuffixImpl();
|
2013-09-13 20:33:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-07 02:03:13 +00:00
|
|
|
void IProfilingBlockInputStream::updateExtremes(Block & block)
|
|
|
|
{
|
2017-12-15 00:01:59 +00:00
|
|
|
size_t num_columns = block.columns();
|
2013-09-07 02:03:13 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
if (!extremes)
|
|
|
|
{
|
2017-12-15 00:01:59 +00:00
|
|
|
MutableColumns extremes_columns(num_columns);
|
2013-09-07 02:03:13 +00:00
|
|
|
|
2017-12-15 00:01:59 +00:00
|
|
|
for (size_t i = 0; i < num_columns; ++i)
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
2017-12-14 21:20:14 +00:00
|
|
|
const ColumnPtr & src = block.safeGetByPosition(i).column;
|
2013-09-07 02:03:13 +00:00
|
|
|
|
2017-12-14 21:20:14 +00:00
|
|
|
if (src->isColumnConst())
|
|
|
|
{
|
|
|
|
/// Equal min and max.
|
|
|
|
extremes_columns[i] = src->cloneResized(2);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Field min_value;
|
|
|
|
Field max_value;
|
2013-09-07 02:03:13 +00:00
|
|
|
|
2017-12-14 21:20:14 +00:00
|
|
|
src->getExtremes(min_value, max_value);
|
2013-09-07 02:03:13 +00:00
|
|
|
|
2017-12-14 21:20:14 +00:00
|
|
|
extremes_columns[i] = src->cloneEmpty();
|
2013-09-07 02:03:13 +00:00
|
|
|
|
2017-12-14 21:20:14 +00:00
|
|
|
extremes_columns[i]->insert(min_value);
|
|
|
|
extremes_columns[i]->insert(max_value);
|
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
2017-12-14 21:20:14 +00:00
|
|
|
|
2017-12-15 00:01:59 +00:00
|
|
|
extremes = block.cloneWithColumns(std::move(extremes_columns));
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-12-15 00:01:59 +00:00
|
|
|
for (size_t i = 0; i < num_columns; ++i)
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
2017-12-14 21:20:14 +00:00
|
|
|
ColumnPtr & old_extremes = extremes.safeGetByPosition(i).column;
|
|
|
|
|
|
|
|
if (old_extremes->isColumnConst())
|
|
|
|
continue;
|
2014-08-22 02:31:54 +00:00
|
|
|
|
2017-12-14 21:20:14 +00:00
|
|
|
Field min_value = (*old_extremes)[0];
|
|
|
|
Field max_value = (*old_extremes)[1];
|
2013-09-07 02:03:13 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
Field cur_min_value;
|
|
|
|
Field cur_max_value;
|
2013-09-07 02:03:13 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
block.safeGetByPosition(i).column->getExtremes(cur_min_value, cur_max_value);
|
2013-09-07 02:03:13 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
if (cur_min_value < min_value)
|
|
|
|
min_value = cur_min_value;
|
|
|
|
if (cur_max_value > max_value)
|
|
|
|
max_value = cur_max_value;
|
2013-09-07 02:03:13 +00:00
|
|
|
|
2017-12-15 00:01:59 +00:00
|
|
|
MutableColumnPtr new_extremes = old_extremes->cloneEmpty();
|
2017-12-14 21:20:14 +00:00
|
|
|
|
|
|
|
new_extremes->insert(min_value);
|
|
|
|
new_extremes->insert(max_value);
|
|
|
|
|
|
|
|
old_extremes = std::move(new_extremes);
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
|
|
|
}
|
2013-09-07 02:03:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-01-12 13:30:57 +00:00
|
|
|
static bool handleOverflowMode(OverflowMode mode, const String & message, int code)
|
2013-09-07 02:03:13 +00:00
|
|
|
{
|
2018-01-12 13:30:57 +00:00
|
|
|
switch (mode)
|
2017-09-08 03:47:27 +00:00
|
|
|
{
|
2018-01-12 13:30:57 +00:00
|
|
|
case OverflowMode::THROW:
|
|
|
|
throw Exception(message, code);
|
|
|
|
case OverflowMode::BREAK:
|
|
|
|
return false;
|
|
|
|
default:
|
|
|
|
throw Exception("Logical error: unknown overflow mode", ErrorCodes::LOGICAL_ERROR);
|
|
|
|
}
|
|
|
|
};
|
2017-09-08 03:47:27 +00:00
|
|
|
|
2018-01-12 13:30:57 +00:00
|
|
|
bool IProfilingBlockInputStream::checkDataSizeLimits()
|
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
if (limits.mode == LIMITS_CURRENT)
|
|
|
|
{
|
|
|
|
/// Check current stream limitations (i.e. max_result_{rows,bytes})
|
|
|
|
|
|
|
|
if (limits.max_rows_to_read && info.rows > limits.max_rows_to_read)
|
2018-01-12 13:30:57 +00:00
|
|
|
return handleOverflowMode(limits.read_overflow_mode,
|
2017-09-08 03:47:27 +00:00
|
|
|
std::string("Limit for result rows")
|
2017-04-01 07:20:54 +00:00
|
|
|
+ " exceeded: read " + toString(info.rows)
|
|
|
|
+ " rows, maximum: " + toString(limits.max_rows_to_read),
|
2017-09-08 03:47:27 +00:00
|
|
|
ErrorCodes::TOO_MUCH_ROWS);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
if (limits.max_bytes_to_read && info.bytes > limits.max_bytes_to_read)
|
2018-01-12 13:30:57 +00:00
|
|
|
return handleOverflowMode(limits.read_overflow_mode,
|
2017-09-08 03:47:27 +00:00
|
|
|
std::string("Limit for result bytes (uncompressed)")
|
2017-04-01 07:20:54 +00:00
|
|
|
+ " exceeded: read " + toString(info.bytes)
|
|
|
|
+ " bytes, maximum: " + toString(limits.max_bytes_to_read),
|
2017-09-08 03:47:27 +00:00
|
|
|
ErrorCodes::TOO_MUCH_BYTES);
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
|
|
|
|
2018-01-12 13:30:57 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IProfilingBlockInputStream::checkTimeLimits()
|
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
if (limits.max_execution_time != 0
|
|
|
|
&& info.total_stopwatch.elapsed() > static_cast<UInt64>(limits.max_execution_time.totalMicroseconds()) * 1000)
|
2018-01-12 13:30:57 +00:00
|
|
|
return handleOverflowMode(limits.timeout_overflow_mode,
|
2017-09-08 03:47:27 +00:00
|
|
|
"Timeout exceeded: elapsed " + toString(info.total_stopwatch.elapsedSeconds())
|
2017-04-01 07:20:54 +00:00
|
|
|
+ " seconds, maximum: " + toString(limits.max_execution_time.totalMicroseconds() / 1000000.0),
|
|
|
|
ErrorCodes::TIMEOUT_EXCEEDED);
|
|
|
|
|
|
|
|
return true;
|
2013-09-07 02:03:13 +00:00
|
|
|
}
|
2013-09-01 04:55:41 +00:00
|
|
|
|
|
|
|
|
2013-09-07 02:03:13 +00:00
|
|
|
void IProfilingBlockInputStream::checkQuota(Block & block)
|
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
switch (limits.mode)
|
|
|
|
{
|
|
|
|
case LIMITS_TOTAL:
|
|
|
|
/// Checked in `progress` method.
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LIMITS_CURRENT:
|
|
|
|
{
|
2017-08-04 14:00:26 +00:00
|
|
|
time_t current_time = time(nullptr);
|
2017-04-01 07:20:54 +00:00
|
|
|
double total_elapsed = info.total_stopwatch.elapsedSeconds();
|
|
|
|
|
|
|
|
quota->checkAndAddResultRowsBytes(current_time, block.rows(), block.bytes());
|
|
|
|
quota->checkAndAddExecutionTime(current_time, Poco::Timespan((total_elapsed - prev_elapsed) * 1000000.0));
|
|
|
|
|
|
|
|
prev_elapsed = total_elapsed;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw Exception("Logical error: unknown limits mode.", ErrorCodes::LOGICAL_ERROR);
|
|
|
|
}
|
2011-08-27 22:43:31 +00:00
|
|
|
}
|
2012-05-17 19:15:53 +00:00
|
|
|
|
|
|
|
|
2014-10-25 18:33:52 +00:00
|
|
|
void IProfilingBlockInputStream::progressImpl(const Progress & value)
|
2012-05-17 19:15:53 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
if (progress_callback)
|
|
|
|
progress_callback(value);
|
|
|
|
|
|
|
|
if (process_list_elem)
|
|
|
|
{
|
|
|
|
if (!process_list_elem->updateProgressIn(value))
|
|
|
|
cancel();
|
|
|
|
|
|
|
|
/// The total amount of data processed or intended for processing in all leaf sources, possibly on remote servers.
|
|
|
|
|
|
|
|
size_t rows_processed = process_list_elem->progress_in.rows;
|
|
|
|
size_t bytes_processed = process_list_elem->progress_in.bytes;
|
|
|
|
|
|
|
|
size_t total_rows_estimate = std::max(rows_processed, process_list_elem->progress_in.total_rows.load(std::memory_order_relaxed));
|
|
|
|
|
|
|
|
/** Check the restrictions on the amount of data to read, the speed of the query, the quota on the amount of data to read.
|
|
|
|
* NOTE: Maybe it makes sense to have them checked directly in ProcessList?
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (limits.mode == LIMITS_TOTAL
|
|
|
|
&& ((limits.max_rows_to_read && total_rows_estimate > limits.max_rows_to_read)
|
|
|
|
|| (limits.max_bytes_to_read && bytes_processed > limits.max_bytes_to_read)))
|
|
|
|
{
|
2017-09-08 03:47:27 +00:00
|
|
|
switch (limits.read_overflow_mode)
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
2017-09-08 03:47:27 +00:00
|
|
|
case OverflowMode::THROW:
|
|
|
|
{
|
|
|
|
if (limits.max_rows_to_read && total_rows_estimate > limits.max_rows_to_read)
|
|
|
|
throw Exception("Limit for rows to read exceeded: " + toString(total_rows_estimate)
|
|
|
|
+ " rows read (or to read), maximum: " + toString(limits.max_rows_to_read),
|
|
|
|
ErrorCodes::TOO_MUCH_ROWS);
|
|
|
|
else
|
|
|
|
throw Exception("Limit for (uncompressed) bytes to read exceeded: " + toString(bytes_processed)
|
|
|
|
+ " bytes read, maximum: " + toString(limits.max_bytes_to_read),
|
|
|
|
ErrorCodes::TOO_MUCH_BYTES);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case OverflowMode::BREAK:
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
2017-09-08 03:47:27 +00:00
|
|
|
/// For `break`, we will stop only if so many lines were actually read, and not just supposed to be read.
|
|
|
|
if ((limits.max_rows_to_read && rows_processed > limits.max_rows_to_read)
|
|
|
|
|| (limits.max_bytes_to_read && bytes_processed > limits.max_bytes_to_read))
|
|
|
|
{
|
|
|
|
cancel();
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
2017-09-08 03:47:27 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
throw Exception("Logical error: unknown overflow mode", ErrorCodes::LOGICAL_ERROR);
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t total_rows = process_list_elem->progress_in.total_rows;
|
|
|
|
|
|
|
|
if (limits.min_execution_speed || (total_rows && limits.timeout_before_checking_execution_speed != 0))
|
|
|
|
{
|
|
|
|
double total_elapsed = info.total_stopwatch.elapsedSeconds();
|
|
|
|
|
|
|
|
if (total_elapsed > limits.timeout_before_checking_execution_speed.totalMicroseconds() / 1000000.0)
|
|
|
|
{
|
|
|
|
if (limits.min_execution_speed && rows_processed / total_elapsed < limits.min_execution_speed)
|
|
|
|
throw Exception("Query is executing too slow: " + toString(rows_processed / total_elapsed)
|
|
|
|
+ " rows/sec., minimum: " + toString(limits.min_execution_speed),
|
|
|
|
ErrorCodes::TOO_SLOW);
|
|
|
|
|
|
|
|
size_t total_rows = process_list_elem->progress_in.total_rows;
|
|
|
|
|
|
|
|
/// If the predicted execution time is longer than `max_execution_time`.
|
|
|
|
if (limits.max_execution_time != 0 && total_rows)
|
|
|
|
{
|
|
|
|
double estimated_execution_time_seconds = total_elapsed * (static_cast<double>(total_rows) / rows_processed);
|
|
|
|
|
|
|
|
if (estimated_execution_time_seconds > limits.max_execution_time.totalSeconds())
|
|
|
|
throw Exception("Estimated query execution time (" + toString(estimated_execution_time_seconds) + " seconds)"
|
|
|
|
+ " is too long. Maximum: " + toString(limits.max_execution_time.totalSeconds())
|
|
|
|
+ ". Estimated rows to process: " + toString(total_rows),
|
|
|
|
ErrorCodes::TOO_SLOW);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (quota != nullptr && limits.mode == LIMITS_TOTAL)
|
|
|
|
{
|
2017-12-18 04:07:26 +00:00
|
|
|
quota->checkAndAddReadRowsBytes(time(nullptr), value.rows, value.bytes);
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
|
|
|
}
|
2012-05-17 19:15:53 +00:00
|
|
|
}
|
2014-08-22 02:31:54 +00:00
|
|
|
|
2011-08-27 22:43:31 +00:00
|
|
|
|
2012-10-17 19:55:56 +00:00
|
|
|
void IProfilingBlockInputStream::cancel()
|
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
bool old_val = false;
|
|
|
|
if (!is_cancelled.compare_exchange_strong(old_val, true, std::memory_order_seq_cst, std::memory_order_relaxed))
|
|
|
|
return;
|
2012-10-17 19:55:56 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
for (auto & child : children)
|
|
|
|
if (IProfilingBlockInputStream * p_child = dynamic_cast<IProfilingBlockInputStream *>(&*child))
|
|
|
|
p_child->cancel();
|
2012-10-17 19:55:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-09-08 03:47:27 +00:00
|
|
|
void IProfilingBlockInputStream::setProgressCallback(const ProgressCallback & callback)
|
2012-05-09 15:15:45 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
progress_callback = callback;
|
2014-08-22 02:31:54 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
for (auto & child : children)
|
|
|
|
if (IProfilingBlockInputStream * p_child = dynamic_cast<IProfilingBlockInputStream *>(&*child))
|
|
|
|
p_child->setProgressCallback(callback);
|
2012-05-09 08:16:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-06-21 06:06:04 +00:00
|
|
|
void IProfilingBlockInputStream::setProcessListElement(ProcessListElement * elem)
|
2013-11-03 05:32:42 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
process_list_elem = elem;
|
2013-11-03 05:32:42 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
for (auto & child : children)
|
|
|
|
if (IProfilingBlockInputStream * p_child = dynamic_cast<IProfilingBlockInputStream *>(&*child))
|
|
|
|
p_child->setProcessListElement(elem);
|
2013-11-03 05:32:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-08 08:27:06 +00:00
|
|
|
const Block & IProfilingBlockInputStream::getTotals()
|
2013-09-01 04:55:41 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
if (totals)
|
|
|
|
return totals;
|
|
|
|
|
|
|
|
for (auto & child : children)
|
|
|
|
{
|
|
|
|
if (IProfilingBlockInputStream * p_child = dynamic_cast<IProfilingBlockInputStream *>(&*child))
|
|
|
|
{
|
|
|
|
const Block & res = p_child->getTotals();
|
|
|
|
if (res)
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return totals;
|
2013-09-01 04:55:41 +00:00
|
|
|
}
|
|
|
|
|
2013-09-07 02:03:13 +00:00
|
|
|
const Block & IProfilingBlockInputStream::getExtremes() const
|
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
if (extremes)
|
|
|
|
return extremes;
|
|
|
|
|
|
|
|
for (const auto & child : children)
|
|
|
|
{
|
|
|
|
if (const IProfilingBlockInputStream * p_child = dynamic_cast<const IProfilingBlockInputStream *>(&*child))
|
|
|
|
{
|
|
|
|
const Block & res = p_child->getExtremes();
|
|
|
|
if (res)
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return extremes;
|
2013-09-07 02:03:13 +00:00
|
|
|
}
|
|
|
|
|
2015-02-15 06:08:31 +00:00
|
|
|
void IProfilingBlockInputStream::collectTotalRowsApprox()
|
|
|
|
{
|
2017-11-01 13:35:34 +00:00
|
|
|
bool old_val = false;
|
|
|
|
if (!collected_total_rows_approx.compare_exchange_strong(old_val, true))
|
2017-04-01 07:20:54 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
for (auto & child : children)
|
|
|
|
{
|
|
|
|
if (IProfilingBlockInputStream * p_child = dynamic_cast<IProfilingBlockInputStream *>(&*child))
|
|
|
|
{
|
|
|
|
p_child->collectTotalRowsApprox();
|
|
|
|
total_rows_approx += p_child->total_rows_approx;
|
|
|
|
}
|
|
|
|
}
|
2015-02-15 06:08:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void IProfilingBlockInputStream::collectAndSendTotalRowsApprox()
|
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
if (collected_total_rows_approx)
|
|
|
|
return;
|
2015-02-15 06:08:31 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
collectTotalRowsApprox();
|
|
|
|
progressImpl(Progress(0, 0, total_rows_approx));
|
2015-02-15 06:08:31 +00:00
|
|
|
}
|
|
|
|
|
2011-08-27 22:43:31 +00:00
|
|
|
}
|