#pragma once #include #include #include namespace DB { /** A table that represents the union of an arbitrary number of other tables. * All tables must have the same structure. */ class StorageMerge : public ext::shared_ptr_helper, public IStorage { friend struct ext::shared_ptr_helper; public: std::string getName() const override { return "Merge"; } std::string getTableName() const override { return table_name; } std::string getDatabaseName() const override { return database_name; } bool isRemote() const override; /// The check is delayed to the read method. It checks the support of the tables used. bool supportsSampling() const override { return true; } bool supportsPrewhere() const override { return true; } bool supportsFinal() const override { return true; } bool supportsIndexForIn() const override { return true; } /// Consider columns coming from the underlying tables NameAndTypePair getColumn(const String & column_name) const override; bool hasColumn(const String & column_name) const override; QueryProcessingStage::Enum getQueryProcessingStage(const Context &) const override; BlockInputStreams read( const Names & column_names, const SelectQueryInfo & query_info, const Context & context, QueryProcessingStage::Enum processed_stage, size_t max_block_size, unsigned num_streams) override; void drop() override {} void rename(const String & /*new_path_to_db*/, const String & new_database_name, const String & new_table_name) override { table_name = new_table_name; database_name = new_database_name; } /// you need to add and remove columns in the sub-tables manually /// the structure of sub-tables is not checked void alter( const AlterCommands & params, const Context & context, TableStructureWriteLockHolder & table_lock_holder) override; bool mayBenefitFromIndexForIn(const ASTPtr & left_in_operand, const Context & query_context) const override; private: String table_name; String database_name; String source_database; OptimizedRegularExpression table_name_regexp; Context global_context; using StorageListWithLocks = std::list>; StorageListWithLocks getSelectedTables(const String & query_id) const; StorageMerge::StorageListWithLocks getSelectedTables(const ASTPtr & query, bool has_virtual_column, bool get_lock, const String & query_id) const; template StoragePtr getFirstTable(F && predicate) const; DatabaseIteratorPtr getDatabaseIterator(const Context & context) const; protected: StorageMerge( const std::string & database_name_, const std::string & table_name_, const ColumnsDescription & columns_, const String & source_database_, const String & table_name_regexp_, const Context & context_); Block getQueryHeader(const Names & column_names, const SelectQueryInfo & query_info, const Context & context, QueryProcessingStage::Enum processed_stage); BlockInputStreams createSourceStreams(const SelectQueryInfo & query_info, const QueryProcessingStage::Enum & processed_stage, const UInt64 max_block_size, const Block & header, const StoragePtr & storage, const TableStructureReadLockHolder & struct_lock, Names & real_column_names, Context & modified_context, size_t streams_num, bool has_table_virtual_column, bool concat_streams = false); void convertingSourceStream(const Block & header, const Context & context, ASTPtr & query, BlockInputStreamPtr & source_stream, QueryProcessingStage::Enum processed_stage); }; }