mirror of
https://github.com/ClickHouse/ClickHouse.git
synced 2024-12-13 09:52:38 +00:00
81 lines
1.9 KiB
C++
81 lines
1.9 KiB
C++
#pragma once
|
|
|
|
#include <DataStreams/IBlockInputStream.h>
|
|
|
|
|
|
namespace DB
|
|
{
|
|
|
|
/** Initialize another source on the first `read` call, and then use it.
|
|
* This is needed, for example, to read from a table that will be populated
|
|
* after creation of LazyBlockInputStream object, but before the first `read` call.
|
|
*/
|
|
class LazyBlockInputStream : public IBlockInputStream
|
|
{
|
|
public:
|
|
using Generator = std::function<BlockInputStreamPtr()>;
|
|
|
|
LazyBlockInputStream(const Block & header_, Generator generator_)
|
|
: header(header_), generator(std::move(generator_))
|
|
{
|
|
}
|
|
|
|
LazyBlockInputStream(const char * name_, const Block & header_, Generator generator_)
|
|
: name(name_), header(header_), generator(std::move(generator_))
|
|
{
|
|
}
|
|
|
|
String getName() const override { return name; }
|
|
|
|
Block getHeader() const override
|
|
{
|
|
return header;
|
|
}
|
|
|
|
/// We call readPrefix lazily. Suppress default behaviour.
|
|
void readPrefix() override {}
|
|
|
|
protected:
|
|
Block readImpl() override
|
|
{
|
|
if (!input)
|
|
{
|
|
input = generator();
|
|
|
|
if (!input)
|
|
return Block();
|
|
|
|
auto * p_input = dynamic_cast<IBlockInputStream *>(input.get());
|
|
|
|
if (p_input)
|
|
{
|
|
/// They could have been set before, but were not passed into the `input`.
|
|
if (progress_callback)
|
|
p_input->setProgressCallback(progress_callback);
|
|
if (process_list_elem)
|
|
p_input->setProcessListElement(process_list_elem);
|
|
}
|
|
|
|
input->readPrefix();
|
|
|
|
{
|
|
addChild(input);
|
|
|
|
if (isCancelled() && p_input)
|
|
p_input->cancel(is_killed);
|
|
}
|
|
}
|
|
|
|
return input->read();
|
|
}
|
|
|
|
private:
|
|
const char * name = "Lazy";
|
|
Block header;
|
|
Generator generator;
|
|
|
|
BlockInputStreamPtr input;
|
|
};
|
|
|
|
}
|