#include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace DB { namespace ErrorCodes { extern const int SYNTAX_ERROR; } InterpreterShowTablesQuery::InterpreterShowTablesQuery(const ASTPtr & query_ptr_, ContextMutablePtr context_) : WithMutableContext(context_), query_ptr(query_ptr_) { } String InterpreterShowTablesQuery::getRewrittenQuery() { const auto & query = query_ptr->as(); /// SHOW DATABASES if (query.databases) { WriteBufferFromOwnString rewritten_query; rewritten_query << "SELECT name FROM system.databases"; if (!query.like.empty()) { rewritten_query << " WHERE name " << (query.not_like ? "NOT " : "") << (query.case_insensitive_like ? "ILIKE " : "LIKE ") << DB::quote << query.like; } if (query.limit_length) rewritten_query << " LIMIT " << query.limit_length; return rewritten_query.str(); } /// SHOW CLUSTER/CLUSTERS if (query.clusters) { WriteBufferFromOwnString rewritten_query; rewritten_query << "SELECT DISTINCT cluster FROM system.clusters"; if (!query.like.empty()) { rewritten_query << " WHERE cluster " << (query.not_like ? "NOT " : "") << (query.case_insensitive_like ? "ILIKE " : "LIKE ") << DB::quote << query.like; } if (query.limit_length) rewritten_query << " LIMIT " << query.limit_length; return rewritten_query.str(); } else if (query.cluster) { WriteBufferFromOwnString rewritten_query; rewritten_query << "SELECT * FROM system.clusters"; rewritten_query << " WHERE cluster = " << DB::quote << query.cluster_str; return rewritten_query.str(); } /// SHOW SETTINGS if (query.m_settings) { WriteBufferFromOwnString rewritten_query; rewritten_query << "SELECT name, type, value FROM system.settings"; if (query.changed) rewritten_query << " WHERE changed = 1"; if (!query.like.empty()) { rewritten_query << (query.changed ? " AND name " : " WHERE name ") << (query.case_insensitive_like ? "ILIKE " : "LIKE ") << DB::quote << query.like; } return rewritten_query.str(); } if (query.temporary && !query.from.empty()) throw Exception("The `FROM` and `TEMPORARY` cannot be used together in `SHOW TABLES`", ErrorCodes::SYNTAX_ERROR); String database = getContext()->resolveDatabase(query.from); DatabaseCatalog::instance().assertDatabaseExists(database); WriteBufferFromOwnString rewritten_query; rewritten_query << "SELECT name FROM system."; if (query.dictionaries) rewritten_query << "dictionaries "; else rewritten_query << "tables "; rewritten_query << "WHERE "; if (query.temporary) { if (query.dictionaries) throw Exception("Temporary dictionaries are not possible.", ErrorCodes::SYNTAX_ERROR); rewritten_query << "is_temporary"; } else rewritten_query << "database = " << DB::quote << database; if (!query.like.empty()) rewritten_query << " AND name " << (query.not_like ? "NOT " : "") << (query.case_insensitive_like ? "ILIKE " : "LIKE ") << DB::quote << query.like; else if (query.where_expression) rewritten_query << " AND (" << query.where_expression << ")"; if (query.limit_length) rewritten_query << " LIMIT " << query.limit_length; return rewritten_query.str(); } BlockIO InterpreterShowTablesQuery::execute() { const auto & query = query_ptr->as(); if (query.caches) { getContext()->checkAccess(AccessType::SHOW_CACHES); Block sample_block{ColumnWithTypeAndName(std::make_shared(), "Caches")}; MutableColumns res_columns = sample_block.cloneEmptyColumns(); auto caches = FileCacheFactory::instance().getAllByName(); for (const auto & [name, _] : caches) res_columns[0]->insert(name); BlockIO res; size_t num_rows = res_columns[0]->size(); auto source = std::make_shared(sample_block, Chunk(std::move(res_columns), num_rows)); res.pipeline = QueryPipeline(std::move(source)); return res; } return executeQuery(getRewrittenQuery(), getContext(), true); } }