2019-06-03 00:57:33 +00:00
|
|
|
#include "ConnectionParameters.h"
|
|
|
|
#include <fstream>
|
|
|
|
#include <Core/Defines.h>
|
|
|
|
#include <Core/Protocol.h>
|
|
|
|
#include <Core/Types.h>
|
|
|
|
#include <IO/ConnectionTimeouts.h>
|
|
|
|
#include <Poco/Util/AbstractConfiguration.h>
|
2023-09-26 15:50:19 +00:00
|
|
|
#include <Common/SSH/Wrappers.h>
|
2019-06-03 00:57:33 +00:00
|
|
|
#include <Common/Exception.h>
|
2021-03-29 00:43:14 +00:00
|
|
|
#include <Common/isLocalAddress.h>
|
|
|
|
#include <Common/DNSResolver.h>
|
2021-10-02 07:13:14 +00:00
|
|
|
#include <base/scope_guard.h>
|
2020-08-10 21:36:49 +00:00
|
|
|
|
2021-10-27 23:10:39 +00:00
|
|
|
#include <readpassphrase/readpassphrase.h>
|
2019-06-03 00:57:33 +00:00
|
|
|
|
2021-07-11 20:35:29 +00:00
|
|
|
|
2019-06-03 00:57:33 +00:00
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
|
|
|
|
namespace ErrorCodes
|
|
|
|
{
|
|
|
|
extern const int BAD_ARGUMENTS;
|
2023-09-26 15:50:19 +00:00
|
|
|
extern const int SUPPORT_IS_DISABLED;
|
2019-06-03 00:57:33 +00:00
|
|
|
}
|
|
|
|
|
2022-01-17 22:57:05 +00:00
|
|
|
ConnectionParameters::ConnectionParameters(const Poco::Util::AbstractConfiguration & config,
|
|
|
|
std::string connection_host,
|
2022-03-10 19:41:03 +00:00
|
|
|
std::optional<UInt16> connection_port)
|
|
|
|
: host(connection_host)
|
|
|
|
, port(connection_port.value_or(getPortFromConfig(config)))
|
2019-06-03 00:57:33 +00:00
|
|
|
{
|
|
|
|
bool is_secure = config.getBool("secure", false);
|
2023-11-12 07:00:10 +00:00
|
|
|
bool is_clickhouse_cloud = connection_host.ends_with(".clickhouse.cloud") || connection_host.ends_with(".clickhouse-staging.com");
|
2023-11-12 06:45:17 +00:00
|
|
|
security = (is_secure || is_clickhouse_cloud) ? Protocol::Secure::Enable : Protocol::Secure::Disable;
|
2019-06-03 00:57:33 +00:00
|
|
|
|
|
|
|
default_database = config.getString("database", "");
|
2020-04-15 01:12:32 +00:00
|
|
|
|
2019-06-03 00:57:33 +00:00
|
|
|
/// changed the default value to "default" to fix the issue when the user in the prompt is blank
|
|
|
|
user = config.getString("user", "default");
|
2020-04-15 01:12:32 +00:00
|
|
|
|
2023-09-26 15:50:19 +00:00
|
|
|
if (!config.has("ssh-key-file"))
|
2019-06-03 00:57:33 +00:00
|
|
|
{
|
2023-09-26 15:50:19 +00:00
|
|
|
bool password_prompt = false;
|
|
|
|
if (config.getBool("ask-password", false))
|
|
|
|
{
|
|
|
|
if (config.has("password"))
|
|
|
|
throw Exception(ErrorCodes::BAD_ARGUMENTS, "Specified both --password and --ask-password. Remove one of them");
|
2019-06-03 00:57:33 +00:00
|
|
|
password_prompt = true;
|
2023-09-26 15:50:19 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
password = config.getString("password", "");
|
|
|
|
/// if the value of --password is omitted, the password will be set implicitly to "\n"
|
|
|
|
if (password == ASK_PASSWORD)
|
|
|
|
password_prompt = true;
|
|
|
|
}
|
|
|
|
if (password_prompt)
|
|
|
|
{
|
|
|
|
std::string prompt{"Password for user (" + user + "): "};
|
|
|
|
char buf[1000] = {};
|
|
|
|
if (auto * result = readpassphrase(prompt.c_str(), buf, sizeof(buf), 0))
|
|
|
|
password = result;
|
|
|
|
}
|
2019-06-03 00:57:33 +00:00
|
|
|
}
|
2023-09-26 15:50:19 +00:00
|
|
|
else
|
2019-06-03 00:57:33 +00:00
|
|
|
{
|
2023-09-26 15:50:19 +00:00
|
|
|
#if USE_SSL
|
|
|
|
std::string filename = config.getString("ssh-key-file");
|
|
|
|
std::string passphrase;
|
|
|
|
if (config.has("ssh-key-passphrase"))
|
|
|
|
{
|
|
|
|
passphrase = config.getString("ssh-key-passphrase");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::string prompt{"Enter your private key passphrase (leave empty for no passphrase): "};
|
|
|
|
char buf[1000] = {};
|
|
|
|
if (auto * result = readpassphrase(prompt.c_str(), buf, sizeof(buf), 0))
|
|
|
|
passphrase = result;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssh::SSHKey key = ssh::SSHKeyFactory::makePrivateFromFile(filename, passphrase);
|
|
|
|
if (!key.isPrivate())
|
|
|
|
throw Exception(ErrorCodes::BAD_ARGUMENTS, "Found public key in file: {} but expected private", filename);
|
|
|
|
|
|
|
|
ssh_private_key = std::move(key);
|
|
|
|
#else
|
|
|
|
throw Exception(ErrorCodes::SUPPORT_IS_DISABLED, "SSH is disabled, because ClickHouse is built without OpenSSL");
|
|
|
|
#endif
|
2019-06-03 00:57:33 +00:00
|
|
|
}
|
2023-09-26 15:50:19 +00:00
|
|
|
|
2022-08-03 19:44:08 +00:00
|
|
|
quota_key = config.getString("quota_key", "");
|
2020-04-15 01:12:32 +00:00
|
|
|
|
2021-03-29 00:43:14 +00:00
|
|
|
/// By default compression is disabled if address looks like localhost.
|
2023-06-28 09:56:11 +00:00
|
|
|
|
|
|
|
/// Avoid DNS request if the host is "localhost".
|
|
|
|
/// If ClickHouse is run under QEMU-user with a binary for a different architecture,
|
|
|
|
/// and there are all listed startup dependency shared libraries available, but not the runtime dependencies of glibc,
|
|
|
|
/// the glibc cannot open "plugins" for DNS resolving, and the DNS resolution does not work.
|
|
|
|
/// At the same time, I want clickhouse-local to always work, regardless.
|
|
|
|
/// TODO: get rid of glibc, or replace getaddrinfo to c-ares.
|
|
|
|
|
|
|
|
compression = config.getBool("compression", host != "localhost" && !isLocalAddress(DNSResolver::instance().resolveHost(host)))
|
2022-01-17 22:57:05 +00:00
|
|
|
? Protocol::Compression::Enable : Protocol::Compression::Disable;
|
2019-06-03 00:57:33 +00:00
|
|
|
|
|
|
|
timeouts = ConnectionTimeouts(
|
2022-01-17 22:57:05 +00:00
|
|
|
Poco::Timespan(config.getInt("connect_timeout", DBMS_DEFAULT_CONNECT_TIMEOUT_SEC), 0),
|
|
|
|
Poco::Timespan(config.getInt("send_timeout", DBMS_DEFAULT_SEND_TIMEOUT_SEC), 0),
|
|
|
|
Poco::Timespan(config.getInt("receive_timeout", DBMS_DEFAULT_RECEIVE_TIMEOUT_SEC), 0),
|
2023-05-17 11:39:04 +00:00
|
|
|
Poco::Timespan(config.getInt("tcp_keep_alive_timeout", 0), 0),
|
|
|
|
Poco::Timespan(config.getInt("handshake_timeout_ms", DBMS_DEFAULT_RECEIVE_TIMEOUT_SEC * 1000), 0));
|
2022-08-12 13:30:46 +00:00
|
|
|
|
|
|
|
timeouts.sync_request_timeout = Poco::Timespan(config.getInt("sync_request_timeout", DBMS_DEFAULT_SYNC_REQUEST_TIMEOUT_SEC), 0);
|
2022-01-17 22:57:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ConnectionParameters::ConnectionParameters(const Poco::Util::AbstractConfiguration & config)
|
2022-01-27 10:44:39 +00:00
|
|
|
: ConnectionParameters(config, config.getString("host", "localhost"), getPortFromConfig(config))
|
2022-01-27 06:19:02 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-03-10 19:41:03 +00:00
|
|
|
UInt16 ConnectionParameters::getPortFromConfig(const Poco::Util::AbstractConfiguration & config)
|
2022-01-17 22:57:05 +00:00
|
|
|
{
|
|
|
|
bool is_secure = config.getBool("secure", false);
|
2022-01-27 06:19:02 +00:00
|
|
|
return config.getInt("port",
|
|
|
|
config.getInt(is_secure ? "tcp_port_secure" : "tcp_port",
|
|
|
|
is_secure ? DBMS_DEFAULT_SECURE_PORT : DBMS_DEFAULT_PORT));
|
2019-06-03 00:57:33 +00:00
|
|
|
}
|
|
|
|
}
|