#include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace DB { namespace ErrorCodes { extern const int NUMBER_OF_ARGUMENTS_DOESNT_MATCH; } StoragePtr ITableFunctionXDBC::executeImpl(const ASTPtr & ast_function, const Context & context) const { const ASTFunction & args_func = typeid_cast(*ast_function); if (!args_func.arguments) throw Exception("Table function '" + getName() + "' must have arguments.", ErrorCodes::LOGICAL_ERROR); ASTs & args = typeid_cast(*args_func.arguments).children; if (args.size() != 2 && args.size() != 3) throw Exception("Table function '" + getName() + "' requires 2 or 3 arguments: " + getName() + "('DSN', table) or "+getName()+"('DSN', schema, table)", ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH); for (auto i = 0u; i < args.size(); ++i) args[i] = evaluateConstantExpressionOrIdentifierAsLiteral(args[i], context); std::string connection_string = ""; std::string schema_name = ""; std::string table_name = ""; if (args.size() == 3) { connection_string = static_cast(*args[0]).value.safeGet(); schema_name = static_cast(*args[1]).value.safeGet(); table_name = static_cast(*args[2]).value.safeGet(); } else if (args.size() == 2) { connection_string = static_cast(*args[0]).value.safeGet(); table_name = static_cast(*args[1]).value.safeGet(); } const auto & config = context.getConfigRef(); /* Infer external table structure */ BridgeHelperPtr helper = createBridgeHelper(config, context.getSettingsRef().http_receive_timeout.value, connection_string); helper->startBridgeSync(); Poco::URI columns_info_uri = helper->getColumnsInfoURI(); columns_info_uri.addQueryParameter("connection_string", connection_string); if (!schema_name.empty()) columns_info_uri.addQueryParameter("schema", schema_name); columns_info_uri.addQueryParameter("table", table_name); ReadWriteBufferFromHTTP buf(columns_info_uri, Poco::Net::HTTPRequest::HTTP_POST, nullptr); std::string columns_info; readStringBinary(columns_info, buf); NamesAndTypesList columns = NamesAndTypesList::parse(columns_info); auto result = createStorage(table_name, schema_name, table_name, ColumnsDescription{columns}, context, helper); if(!result) throw Exception("Failed to instantiate storage from table function " + getName()); result->startup(); return result; } void registerTableFunctionJDBC(TableFunctionFactory & factory) { factory.registerFunction(); } void registerTableFunctionIDBC(TableFunctionFactory & factory) { factory.registerFunction(); } }