#pragma once #include 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 IProfilingBlockInputStream { public: using Generator = std::function; 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; } protected: Block readImpl() override { if (!input) { input = generator(); if (!input) return Block(); auto * p_input = dynamic_cast(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(); } } return input->read(); } private: const char * name = "Lazy"; Block header; Generator generator; BlockInputStreamPtr input; }; }