2011-08-27 22:43:31 +00:00
|
|
|
|
#include <iomanip>
|
|
|
|
|
|
2012-02-27 07:54:16 +00:00
|
|
|
|
/*#include <Poco/Mutex.h>
|
|
|
|
|
#include <Poco/Ext/ThreadNumber.h>*/
|
|
|
|
|
|
2013-09-07 02:03:13 +00:00
|
|
|
|
#include <DB/Columns/ColumnConst.h>
|
2011-09-04 21:23:19 +00:00
|
|
|
|
#include <DB/DataStreams/IProfilingBlockInputStream.h>
|
2011-08-27 22:43:31 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
namespace DB
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
2013-05-22 14:57:43 +00:00
|
|
|
|
void BlockStreamProfileInfo::read(ReadBuffer & in)
|
|
|
|
|
{
|
|
|
|
|
readVarUInt(rows, in);
|
|
|
|
|
readVarUInt(blocks, in);
|
|
|
|
|
readVarUInt(bytes, in);
|
|
|
|
|
readBinary(applied_limit, in);
|
|
|
|
|
readVarUInt(rows_before_limit, in);
|
|
|
|
|
readBinary(calculated_rows_before_limit, in);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void BlockStreamProfileInfo::write(WriteBuffer & out) const
|
|
|
|
|
{
|
|
|
|
|
writeVarUInt(rows, out);
|
|
|
|
|
writeVarUInt(blocks, out);
|
|
|
|
|
writeVarUInt(bytes, out);
|
|
|
|
|
writeBinary(hasAppliedLimit(), out);
|
|
|
|
|
writeVarUInt(getRowsBeforeLimit(), out);
|
|
|
|
|
writeBinary(calculated_rows_before_limit, out);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
size_t BlockStreamProfileInfo::getRowsBeforeLimit() const
|
|
|
|
|
{
|
|
|
|
|
if (!calculated_rows_before_limit)
|
|
|
|
|
calculateRowsBeforeLimit();
|
|
|
|
|
return rows_before_limit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool BlockStreamProfileInfo::hasAppliedLimit() const
|
|
|
|
|
{
|
|
|
|
|
if (!calculated_rows_before_limit)
|
|
|
|
|
calculateRowsBeforeLimit();
|
|
|
|
|
return applied_limit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2012-05-17 19:15:53 +00:00
|
|
|
|
void BlockStreamProfileInfo::update(Block & block)
|
2011-08-27 22:43:31 +00:00
|
|
|
|
{
|
|
|
|
|
++blocks;
|
|
|
|
|
rows += block.rows();
|
2012-05-17 19:15:53 +00:00
|
|
|
|
bytes += block.bytes();
|
2011-09-05 00:51:25 +00:00
|
|
|
|
|
|
|
|
|
if (column_names.empty())
|
|
|
|
|
column_names = block.dumpNames();
|
2011-08-27 22:43:31 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-19 00:08:49 +00:00
|
|
|
|
void BlockStreamProfileInfo::collectInfosForStreamsWithName(const String & name, BlockStreamProfileInfos & res) const
|
2013-05-20 12:21:51 +00:00
|
|
|
|
{
|
2013-09-19 00:08:49 +00:00
|
|
|
|
if (stream_name == name)
|
2013-05-22 14:57:43 +00:00
|
|
|
|
{
|
2013-09-19 00:08:49 +00:00
|
|
|
|
res.push_back(this);
|
|
|
|
|
return;
|
2013-05-22 14:57:43 +00:00
|
|
|
|
}
|
2013-09-19 00:08:49 +00:00
|
|
|
|
|
|
|
|
|
for (BlockStreamProfileInfos::const_iterator it = nested_infos.begin(); it != nested_infos.end(); ++it)
|
|
|
|
|
(*it)->collectInfosForStreamsWithName(name, res);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void BlockStreamProfileInfo::calculateRowsBeforeLimit() const
|
|
|
|
|
{
|
|
|
|
|
calculated_rows_before_limit = true;
|
|
|
|
|
|
|
|
|
|
/// есть ли Limit?
|
|
|
|
|
BlockStreamProfileInfos limits;
|
|
|
|
|
collectInfosForStreamsWithName("Limit", limits);
|
|
|
|
|
if (limits.empty())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
applied_limit = true;
|
|
|
|
|
|
|
|
|
|
/** Берём количество строчек, прочитанных ниже PartialSorting-а, если есть, или ниже Limit-а.
|
|
|
|
|
* Это нужно, потому что сортировка может вернуть только часть строк.
|
|
|
|
|
*/
|
|
|
|
|
BlockStreamProfileInfos partial_sortings;
|
|
|
|
|
collectInfosForStreamsWithName("PartialSorting", partial_sortings);
|
|
|
|
|
|
|
|
|
|
BlockStreamProfileInfos & limits_or_sortings = partial_sortings.empty() ? limits : partial_sortings;
|
|
|
|
|
|
|
|
|
|
for (BlockStreamProfileInfos::const_iterator it = limits_or_sortings.begin(); it != limits_or_sortings.end(); ++it)
|
|
|
|
|
for (BlockStreamProfileInfos::const_iterator jt = (*it)->nested_infos.begin(); jt != (*it)->nested_infos.end(); ++jt)
|
|
|
|
|
rows_before_limit += (*jt)->rows;
|
2013-05-20 12:21:51 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2011-08-27 22:43:31 +00:00
|
|
|
|
void BlockStreamProfileInfo::print(std::ostream & ostr) const
|
|
|
|
|
{
|
2012-12-25 17:25:44 +00:00
|
|
|
|
UInt64 elapsed = work_stopwatch.elapsed();
|
|
|
|
|
UInt64 nested_elapsed = 0;
|
|
|
|
|
double elapsed_seconds = work_stopwatch.elapsedSeconds();
|
|
|
|
|
double nested_elapsed_seconds = 0;
|
|
|
|
|
|
2011-09-26 01:50:32 +00:00
|
|
|
|
UInt64 nested_rows = 0;
|
|
|
|
|
UInt64 nested_blocks = 0;
|
|
|
|
|
UInt64 nested_bytes = 0;
|
|
|
|
|
|
|
|
|
|
if (!nested_infos.empty())
|
|
|
|
|
{
|
|
|
|
|
for (BlockStreamProfileInfos::const_iterator it = nested_infos.begin(); it != nested_infos.end(); ++it)
|
|
|
|
|
{
|
|
|
|
|
if ((*it)->work_stopwatch.elapsed() > nested_elapsed)
|
2012-12-25 17:25:44 +00:00
|
|
|
|
{
|
2011-09-26 01:50:32 +00:00
|
|
|
|
nested_elapsed = (*it)->work_stopwatch.elapsed();
|
2012-12-25 17:25:44 +00:00
|
|
|
|
nested_elapsed_seconds = (*it)->work_stopwatch.elapsedSeconds();
|
|
|
|
|
}
|
|
|
|
|
|
2011-09-26 01:50:32 +00:00
|
|
|
|
nested_rows += (*it)->rows;
|
|
|
|
|
nested_blocks += (*it)->blocks;
|
|
|
|
|
nested_bytes += (*it)->bytes;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ostr << std::fixed << std::setprecision(2)
|
|
|
|
|
<< "Columns: " << column_names << std::endl
|
2012-12-25 17:25:44 +00:00
|
|
|
|
<< "Elapsed: " << elapsed_seconds << " sec. "
|
|
|
|
|
<< "(" << elapsed * 100.0 / total_stopwatch.elapsed() << "%), " << std::endl;
|
2011-09-26 01:50:32 +00:00
|
|
|
|
|
|
|
|
|
if (!nested_infos.empty())
|
2011-09-26 04:26:22 +00:00
|
|
|
|
{
|
2012-12-25 17:25:44 +00:00
|
|
|
|
double self_percents = (elapsed - nested_elapsed) * 100.0 / total_stopwatch.elapsed();
|
2011-09-26 04:26:22 +00:00
|
|
|
|
|
2012-12-25 17:25:44 +00:00
|
|
|
|
ostr<< "Elapsed (self): " << (elapsed_seconds - nested_elapsed_seconds) << " sec. "
|
2011-09-26 04:26:22 +00:00
|
|
|
|
<< "(" << (self_percents >= 50 ? "\033[1;31m" : (self_percents >= 10 ? "\033[1;33m" : "")) /// Раскраска больших значений
|
|
|
|
|
<< self_percents << "%"
|
|
|
|
|
<< (self_percents >= 10 ? "\033[0m" : "") << "), " << std::endl
|
2012-12-25 17:25:44 +00:00
|
|
|
|
<< "Rows (in): " << nested_rows << ", per second: " << nested_rows / elapsed_seconds << ", " << std::endl
|
|
|
|
|
<< "Blocks (in): " << nested_blocks << ", per second: " << nested_blocks / elapsed_seconds << ", " << std::endl
|
2011-09-26 01:50:32 +00:00
|
|
|
|
<< " " << nested_bytes / 1000000.0 << " MB (memory), "
|
2012-12-25 17:25:44 +00:00
|
|
|
|
<< nested_bytes * 1000 / elapsed << " MB/s (memory), " << std::endl;
|
2011-09-28 06:56:22 +00:00
|
|
|
|
|
|
|
|
|
if (self_percents > 0.1)
|
2012-12-25 17:25:44 +00:00
|
|
|
|
ostr << "Rows per second (in, self): " << (nested_rows / (elapsed_seconds - nested_elapsed_seconds))
|
|
|
|
|
<< ", " << (elapsed - nested_elapsed) / nested_rows << " ns/row, " << std::endl;
|
2011-09-26 04:26:22 +00:00
|
|
|
|
}
|
2011-09-26 01:50:32 +00:00
|
|
|
|
|
2012-12-25 17:25:44 +00:00
|
|
|
|
ostr << "Rows (out): " << rows << ", per second: " << rows / elapsed_seconds << ", " << std::endl
|
|
|
|
|
<< "Blocks (out): " << blocks << ", per second: " << blocks / elapsed_seconds << ", " << std::endl
|
|
|
|
|
<< " " << bytes / 1000000.0 << " MB (memory), " << bytes * 1000 / elapsed << " MB/s (memory), " << std::endl
|
2011-09-26 01:50:32 +00:00
|
|
|
|
<< "Average block size (out): " << rows / blocks << "." << std::endl;
|
2011-08-27 22:43:31 +00:00
|
|
|
|
}
|
|
|
|
|
|
2012-08-23 23:49:28 +00:00
|
|
|
|
|
2011-09-04 21:23:19 +00:00
|
|
|
|
Block IProfilingBlockInputStream::read()
|
2011-08-27 22:43:31 +00:00
|
|
|
|
{
|
|
|
|
|
if (!info.started)
|
2011-09-26 01:50:32 +00:00
|
|
|
|
{
|
2011-08-27 22:43:31 +00:00
|
|
|
|
info.total_stopwatch.start();
|
2013-05-22 12:10:33 +00:00
|
|
|
|
info.stream_name = getShortName();
|
2011-09-26 01:50:32 +00:00
|
|
|
|
|
|
|
|
|
for (BlockInputStreams::const_iterator it = children.begin(); it != children.end(); ++it)
|
2012-05-09 08:16:09 +00:00
|
|
|
|
if (const IProfilingBlockInputStream * child = dynamic_cast<const IProfilingBlockInputStream *>(&**it))
|
|
|
|
|
info.nested_infos.push_back(&child->info);
|
2011-09-26 01:50:32 +00:00
|
|
|
|
|
|
|
|
|
info.started = true;
|
|
|
|
|
}
|
2012-05-09 08:16:09 +00:00
|
|
|
|
|
2013-11-17 19:14:17 +00:00
|
|
|
|
Block res;
|
|
|
|
|
|
2012-10-20 05:54:35 +00:00
|
|
|
|
if (is_cancelled)
|
2013-11-17 19:14:17 +00:00
|
|
|
|
return res;
|
2012-05-31 00:47:05 +00:00
|
|
|
|
|
2011-08-27 22:43:31 +00:00
|
|
|
|
info.work_stopwatch.start();
|
2014-01-08 16:23:31 +00:00
|
|
|
|
res = readImpl();
|
2011-08-27 22:43:31 +00:00
|
|
|
|
info.work_stopwatch.stop();
|
|
|
|
|
|
2011-09-25 05:07:47 +00:00
|
|
|
|
/* if (res)
|
2011-09-25 03:37:09 +00:00
|
|
|
|
{
|
2012-02-27 07:54:16 +00:00
|
|
|
|
static Poco::FastMutex mutex;
|
|
|
|
|
Poco::ScopedLock<Poco::FastMutex> lock(mutex);
|
2012-03-05 07:58:34 +00:00
|
|
|
|
|
2011-09-25 03:37:09 +00:00
|
|
|
|
std::cerr << std::endl;
|
2012-06-24 23:17:06 +00:00
|
|
|
|
std::cerr << "[ " << Poco::ThreadNumber::get() << " ]\t" << getShortName() << std::endl;
|
2012-03-05 07:58:34 +00:00
|
|
|
|
std::cerr << "[ " << Poco::ThreadNumber::get() << " ]\t";
|
|
|
|
|
|
|
|
|
|
for (size_t i = 0; i < res.columns(); ++i)
|
|
|
|
|
{
|
|
|
|
|
if (i != 0)
|
|
|
|
|
std::cerr << ", ";
|
|
|
|
|
std::cerr << res.getByPosition(i).name << " (" << res.getByPosition(i).column->size() << ")";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::cerr << std::endl;
|
2011-09-25 05:07:47 +00:00
|
|
|
|
}*/
|
2012-05-09 15:15:45 +00:00
|
|
|
|
|
2012-03-05 07:58:34 +00:00
|
|
|
|
if (res)
|
2013-09-07 02:03:13 +00:00
|
|
|
|
{
|
2012-05-17 19:15:53 +00:00
|
|
|
|
info.update(res);
|
2013-09-07 02:03:13 +00:00
|
|
|
|
|
|
|
|
|
if (enabled_extremes)
|
|
|
|
|
updateExtremes(res);
|
|
|
|
|
|
|
|
|
|
if (!checkLimits())
|
2013-11-17 19:14:17 +00:00
|
|
|
|
{
|
|
|
|
|
res.clear();
|
|
|
|
|
return res;
|
|
|
|
|
}
|
2013-09-07 02:03:13 +00:00
|
|
|
|
|
2014-04-08 07:31:51 +00:00
|
|
|
|
if (quota != nullptr)
|
2013-09-07 02:03:13 +00:00
|
|
|
|
checkQuota(res);
|
|
|
|
|
}
|
2012-10-17 19:59:27 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/** Если поток закончился, то ещё попросим всех детей прервать выполнение.
|
|
|
|
|
* Это имеет смысл при выполнении запроса с LIMIT-ом:
|
|
|
|
|
* - бывает ситуация, когда все необходимые данные уже прочитали,
|
|
|
|
|
* но источники-дети ещё продолжают работать,
|
|
|
|
|
* при чём они могут работать в отдельных потоках или даже удалённо.
|
|
|
|
|
*/
|
|
|
|
|
cancel();
|
|
|
|
|
}
|
2011-09-25 03:37:09 +00:00
|
|
|
|
|
2013-11-03 05:32:42 +00:00
|
|
|
|
progress(res.rows(), res.bytes());
|
2012-12-25 17:25:44 +00:00
|
|
|
|
|
2013-09-07 02:03:13 +00:00
|
|
|
|
return res;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-13 20:33:09 +00:00
|
|
|
|
void IProfilingBlockInputStream::readSuffix()
|
|
|
|
|
{
|
|
|
|
|
for (BlockInputStreams::iterator it = children.begin(); it != children.end(); ++it)
|
|
|
|
|
(*it)->readSuffix();
|
|
|
|
|
|
|
|
|
|
readSuffixImpl();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-07 02:03:13 +00:00
|
|
|
|
void IProfilingBlockInputStream::updateExtremes(Block & block)
|
|
|
|
|
{
|
|
|
|
|
size_t columns = block.columns();
|
|
|
|
|
|
|
|
|
|
if (!extremes)
|
|
|
|
|
{
|
|
|
|
|
extremes = block.cloneEmpty();
|
|
|
|
|
|
|
|
|
|
for (size_t i = 0; i < columns; ++i)
|
|
|
|
|
{
|
|
|
|
|
Field min_value;
|
|
|
|
|
Field max_value;
|
|
|
|
|
|
|
|
|
|
block.getByPosition(i).column->getExtremes(min_value, max_value);
|
|
|
|
|
|
|
|
|
|
ColumnPtr & column = extremes.getByPosition(i).column;
|
|
|
|
|
|
|
|
|
|
if (column->isConst())
|
|
|
|
|
column = dynamic_cast<const IColumnConst &>(*column).convertToFullColumn();
|
|
|
|
|
|
|
|
|
|
column->insert(min_value);
|
|
|
|
|
column->insert(max_value);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
for (size_t i = 0; i < columns; ++i)
|
|
|
|
|
{
|
|
|
|
|
ColumnPtr & column = extremes.getByPosition(i).column;
|
|
|
|
|
|
|
|
|
|
Field min_value = (*column)[0];
|
|
|
|
|
Field max_value = (*column)[1];
|
|
|
|
|
|
|
|
|
|
Field cur_min_value;
|
|
|
|
|
Field cur_max_value;
|
|
|
|
|
|
|
|
|
|
block.getByPosition(i).column->getExtremes(cur_min_value, cur_max_value);
|
|
|
|
|
|
|
|
|
|
if (cur_min_value < min_value)
|
|
|
|
|
min_value = cur_min_value;
|
|
|
|
|
if (cur_max_value > max_value)
|
|
|
|
|
max_value = cur_max_value;
|
|
|
|
|
|
|
|
|
|
column = column->cloneEmpty();
|
|
|
|
|
column->insert(min_value);
|
|
|
|
|
column->insert(max_value);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool IProfilingBlockInputStream::checkLimits()
|
|
|
|
|
{
|
2012-12-25 17:25:44 +00:00
|
|
|
|
/// Проверка ограничений.
|
2012-12-25 17:27:18 +00:00
|
|
|
|
if ((limits.max_rows_to_read && info.rows > limits.max_rows_to_read)
|
|
|
|
|
|| (limits.max_bytes_to_read && info.bytes > limits.max_bytes_to_read))
|
2012-12-25 17:25:44 +00:00
|
|
|
|
{
|
2014-02-17 23:56:45 +00:00
|
|
|
|
if (limits.read_overflow_mode == OverflowMode::THROW)
|
2013-11-12 23:32:04 +00:00
|
|
|
|
throw Exception(std::string("Limit for ")
|
|
|
|
|
+ (limits.mode == LIMITS_CURRENT ? "result rows" : "rows to read")
|
|
|
|
|
+ " exceeded: read " + toString(info.rows)
|
2013-06-21 20:34:19 +00:00
|
|
|
|
+ " rows, maximum: " + toString(limits.max_rows_to_read),
|
2012-12-25 17:25:44 +00:00
|
|
|
|
ErrorCodes::TOO_MUCH_ROWS);
|
|
|
|
|
|
2014-02-17 23:56:45 +00:00
|
|
|
|
if (limits.read_overflow_mode == OverflowMode::BREAK)
|
2013-09-07 02:03:13 +00:00
|
|
|
|
return false;
|
2012-12-25 17:25:44 +00:00
|
|
|
|
|
2013-05-17 08:02:34 +00:00
|
|
|
|
throw Exception("Logical error: unknown overflow mode", ErrorCodes::LOGICAL_ERROR);
|
2012-12-25 17:25:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (limits.max_execution_time != 0
|
|
|
|
|
&& info.total_stopwatch.elapsed() > static_cast<UInt64>(limits.max_execution_time.totalMicroseconds()) * 1000)
|
|
|
|
|
{
|
2014-02-17 23:56:45 +00:00
|
|
|
|
if (limits.timeout_overflow_mode == OverflowMode::THROW)
|
2013-06-21 20:34:19 +00:00
|
|
|
|
throw Exception("Timeout exceeded: elapsed " + toString(info.total_stopwatch.elapsedSeconds())
|
|
|
|
|
+ " seconds, maximum: " + toString(limits.max_execution_time.totalMicroseconds() / 1000000.0),
|
2012-12-25 17:25:44 +00:00
|
|
|
|
ErrorCodes::TIMEOUT_EXCEEDED);
|
|
|
|
|
|
2014-02-17 23:56:45 +00:00
|
|
|
|
if (limits.timeout_overflow_mode == OverflowMode::BREAK)
|
2013-09-07 02:03:13 +00:00
|
|
|
|
return false;
|
2012-12-25 17:25:44 +00:00
|
|
|
|
|
2013-05-17 08:02:34 +00:00
|
|
|
|
throw Exception("Logical error: unknown overflow mode", ErrorCodes::LOGICAL_ERROR);
|
2012-12-25 17:25:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
2013-09-07 02:03:13 +00:00
|
|
|
|
return true;
|
|
|
|
|
}
|
2013-09-01 04:55:41 +00:00
|
|
|
|
|
|
|
|
|
|
2013-09-07 02:03:13 +00:00
|
|
|
|
void IProfilingBlockInputStream::checkQuota(Block & block)
|
|
|
|
|
{
|
|
|
|
|
time_t current_time = time(0);
|
|
|
|
|
double total_elapsed = info.total_stopwatch.elapsedSeconds();
|
2013-09-01 04:55:41 +00:00
|
|
|
|
|
2013-11-12 23:15:09 +00:00
|
|
|
|
switch (limits.mode)
|
2013-09-07 02:03:13 +00:00
|
|
|
|
{
|
2013-11-12 23:15:09 +00:00
|
|
|
|
case LIMITS_TOTAL:
|
2013-11-03 05:32:42 +00:00
|
|
|
|
/// Проверяется в методе progress.
|
2013-09-07 02:03:13 +00:00
|
|
|
|
break;
|
|
|
|
|
|
2013-11-12 23:15:09 +00:00
|
|
|
|
case LIMITS_CURRENT:
|
2013-09-07 02:03:13 +00:00
|
|
|
|
quota->checkAndAddResultRowsBytes(current_time, block.rows(), block.bytes());
|
|
|
|
|
quota->checkAndAddExecutionTime(current_time, Poco::Timespan((total_elapsed - prev_elapsed) * 1000000.0));
|
|
|
|
|
break;
|
2013-09-01 04:55:41 +00:00
|
|
|
|
|
2013-09-07 02:03:13 +00:00
|
|
|
|
default:
|
2013-11-12 23:15:09 +00:00
|
|
|
|
throw Exception("Logical error: unknown limits mode.", ErrorCodes::LOGICAL_ERROR);
|
2013-09-01 04:55:41 +00:00
|
|
|
|
}
|
2013-09-07 02:03:13 +00:00
|
|
|
|
|
|
|
|
|
prev_elapsed = total_elapsed;
|
2011-08-27 22:43:31 +00:00
|
|
|
|
}
|
2012-05-17 19:15:53 +00:00
|
|
|
|
|
|
|
|
|
|
2013-11-03 05:32:42 +00:00
|
|
|
|
void IProfilingBlockInputStream::progressImpl(size_t rows, size_t bytes)
|
2012-05-17 19:15:53 +00:00
|
|
|
|
{
|
2013-11-03 05:32:42 +00:00
|
|
|
|
/// Данные для прогресса берутся из листовых источников.
|
|
|
|
|
if (children.empty())
|
|
|
|
|
{
|
|
|
|
|
if (progress_callback)
|
|
|
|
|
progress_callback(rows, bytes);
|
|
|
|
|
|
|
|
|
|
if (process_list_elem)
|
|
|
|
|
{
|
2014-02-12 17:31:02 +00:00
|
|
|
|
if (!process_list_elem->update(rows, bytes))
|
|
|
|
|
cancel();
|
2013-11-03 05:32:42 +00:00
|
|
|
|
|
|
|
|
|
/// Общее количество данных, обработанных во всех листовых источниках, возможно, на удалённых серверах.
|
|
|
|
|
|
|
|
|
|
size_t total_rows = process_list_elem->rows_processed;
|
|
|
|
|
size_t total_bytes = process_list_elem->bytes_processed;
|
|
|
|
|
double total_elapsed = info.total_stopwatch.elapsedSeconds();
|
|
|
|
|
|
|
|
|
|
/** Проверяем ограничения на объём данных для чтения, скорость выполнения запроса, квоту на объём данных для чтения.
|
|
|
|
|
* NOTE: Может быть, имеет смысл сделать, чтобы они проверялись прямо в ProcessList?
|
|
|
|
|
*/
|
|
|
|
|
|
2013-11-12 23:15:09 +00:00
|
|
|
|
if (limits.mode == LIMITS_TOTAL
|
|
|
|
|
&& ((limits.max_rows_to_read && total_rows > limits.max_rows_to_read)
|
|
|
|
|
|| (limits.max_bytes_to_read && total_bytes > limits.max_bytes_to_read)))
|
2013-11-03 05:32:42 +00:00
|
|
|
|
{
|
2014-02-17 23:56:45 +00:00
|
|
|
|
if (limits.read_overflow_mode == OverflowMode::THROW)
|
2013-11-03 05:32:42 +00:00
|
|
|
|
throw Exception("Limit for rows to read exceeded: read " + toString(total_rows)
|
|
|
|
|
+ " rows, maximum: " + toString(limits.max_rows_to_read),
|
|
|
|
|
ErrorCodes::TOO_MUCH_ROWS);
|
2014-02-17 23:56:45 +00:00
|
|
|
|
else if (limits.read_overflow_mode == OverflowMode::BREAK)
|
2013-11-03 05:32:42 +00:00
|
|
|
|
cancel();
|
|
|
|
|
else
|
|
|
|
|
throw Exception("Logical error: unknown overflow mode", ErrorCodes::LOGICAL_ERROR);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (limits.min_execution_speed
|
|
|
|
|
&& total_elapsed > limits.timeout_before_checking_execution_speed.totalMicroseconds() / 1000000.0
|
|
|
|
|
&& total_rows / total_elapsed < limits.min_execution_speed)
|
|
|
|
|
{
|
|
|
|
|
throw Exception("Query is executing too slow: " + toString(total_rows / total_elapsed)
|
|
|
|
|
+ " rows/sec., minimum: " + toString(limits.min_execution_speed),
|
|
|
|
|
ErrorCodes::TOO_SLOW);
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-08 07:31:51 +00:00
|
|
|
|
if (quota != nullptr && limits.mode == LIMITS_TOTAL)
|
2013-11-03 05:32:42 +00:00
|
|
|
|
{
|
|
|
|
|
quota->checkAndAddReadRowsBytes(time(0), rows, bytes);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2012-05-17 19:15:53 +00:00
|
|
|
|
}
|
2011-08-27 22:43:31 +00:00
|
|
|
|
|
|
|
|
|
|
2011-09-04 21:23:19 +00:00
|
|
|
|
const BlockStreamProfileInfo & IProfilingBlockInputStream::getInfo() const
|
2011-08-27 22:43:31 +00:00
|
|
|
|
{
|
|
|
|
|
return info;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2012-10-17 19:55:56 +00:00
|
|
|
|
void IProfilingBlockInputStream::cancel()
|
|
|
|
|
{
|
2012-11-10 05:13:46 +00:00
|
|
|
|
if (!__sync_bool_compare_and_swap(&is_cancelled, false, true))
|
|
|
|
|
return;
|
2012-10-17 19:55:56 +00:00
|
|
|
|
|
|
|
|
|
for (BlockInputStreams::iterator it = children.begin(); it != children.end(); ++it)
|
|
|
|
|
if (IProfilingBlockInputStream * child = dynamic_cast<IProfilingBlockInputStream *>(&**it))
|
|
|
|
|
child->cancel();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2012-05-09 15:15:45 +00:00
|
|
|
|
void IProfilingBlockInputStream::setProgressCallback(ProgressCallback callback)
|
|
|
|
|
{
|
2012-05-17 19:15:53 +00:00
|
|
|
|
progress_callback = callback;
|
|
|
|
|
|
|
|
|
|
for (BlockInputStreams::iterator it = children.begin(); it != children.end(); ++it)
|
|
|
|
|
if (IProfilingBlockInputStream * child = dynamic_cast<IProfilingBlockInputStream *>(&**it))
|
|
|
|
|
child->setProgressCallback(callback);
|
2012-05-09 08:16:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-11-03 05:32:42 +00:00
|
|
|
|
void IProfilingBlockInputStream::setProcessListElement(ProcessList::Element * elem)
|
|
|
|
|
{
|
|
|
|
|
process_list_elem = elem;
|
|
|
|
|
|
|
|
|
|
for (BlockInputStreams::iterator it = children.begin(); it != children.end(); ++it)
|
|
|
|
|
if (IProfilingBlockInputStream * child = dynamic_cast<IProfilingBlockInputStream *>(&**it))
|
|
|
|
|
child->setProcessListElement(elem);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-08 08:27:06 +00:00
|
|
|
|
const Block & IProfilingBlockInputStream::getTotals()
|
2013-09-01 04:55:41 +00:00
|
|
|
|
{
|
|
|
|
|
if (totals)
|
|
|
|
|
return totals;
|
|
|
|
|
|
2013-09-08 08:27:06 +00:00
|
|
|
|
for (BlockInputStreams::iterator it = children.begin(); it != children.end(); ++it)
|
2013-09-01 04:55:41 +00:00
|
|
|
|
{
|
2013-09-08 08:27:06 +00:00
|
|
|
|
if (IProfilingBlockInputStream * child = dynamic_cast<IProfilingBlockInputStream *>(&**it))
|
2013-09-01 04:55:41 +00:00
|
|
|
|
{
|
|
|
|
|
const Block & res = child->getTotals();
|
|
|
|
|
if (res)
|
|
|
|
|
return res;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return totals;
|
|
|
|
|
}
|
|
|
|
|
|
2013-09-07 02:03:13 +00:00
|
|
|
|
const Block & IProfilingBlockInputStream::getExtremes() const
|
|
|
|
|
{
|
|
|
|
|
if (extremes)
|
|
|
|
|
return extremes;
|
|
|
|
|
|
|
|
|
|
for (BlockInputStreams::const_iterator it = children.begin(); it != children.end(); ++it)
|
|
|
|
|
{
|
|
|
|
|
if (const IProfilingBlockInputStream * child = dynamic_cast<const IProfilingBlockInputStream *>(&**it))
|
|
|
|
|
{
|
|
|
|
|
const Block & res = child->getExtremes();
|
|
|
|
|
if (res)
|
|
|
|
|
return res;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return extremes;
|
|
|
|
|
}
|
|
|
|
|
|
2013-09-01 04:55:41 +00:00
|
|
|
|
|
2011-08-27 22:43:31 +00:00
|
|
|
|
}
|