/*
* Copyright (C) 2019 ScyllaDB
*/
/*
* This file is part of Scylla.
*
* Scylla is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Scylla is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Scylla. If not, see .
*/
#pragma once
#include "db/query_context.hh"
#include
#include
#include "seastarx.hh"
#include
namespace db::system_keyspace {
extern const char *const CLIENTS;
}
enum class client_type {
cql = 0,
thrift,
alternator,
};
sstring to_string(client_type ct);
enum class changed_column {
username = 0,
connection_stage,
driver_name,
driver_version,
hostname,
protocol_version,
};
template constexpr const char* column_literal = "";
template <> inline constexpr const char* column_literal = "username";
template <> inline constexpr const char* column_literal = "connection_stage";
template <> inline constexpr const char* column_literal = "driver_name";
template <> inline constexpr const char* column_literal = "driver_version";
template <> inline constexpr const char* column_literal = "hostname";
template <> inline constexpr const char* column_literal = "protocol_version";
enum class client_connection_stage {
established = 0,
authenticating,
ready,
};
template constexpr const char* connection_stage_literal = "";
template <> inline constexpr const char* connection_stage_literal = "ESTABLISHED";
template <> inline constexpr const char* connection_stage_literal = "AUTHENTICATING";
template <> inline constexpr const char* connection_stage_literal = "READY";
// Representation of a row in `system.clients'. std::optionals are for nullable cells.
struct client_data {
net::inet_address ip;
int32_t port;
client_type ct;
client_connection_stage connection_stage = client_connection_stage::established;
int32_t shard_id; /// ID of server-side shard which is processing the connection.
// `optional' column means that it's nullable (possibly because it's
// unimplemented yet). If you want to fill ("implement") any of them,
// remember to update the query in `notify_new_client()'.
std::optional driver_name;
std::optional driver_version;
std::optional hostname;
std::optional protocol_version;
std::optional ssl_cipher_suite;
std::optional ssl_enabled;
std::optional ssl_protocol;
std::optional username;
};
future<> notify_new_client(client_data cd);
future<> notify_disconnected_client(net::inet_address addr, int port, client_type ct);
future<> clear_clientlist();
template
struct notify_client_change {
template
future<> operator()(net::inet_address addr, int port, client_type ct, T&& value) {
const static sstring req
= format("UPDATE system.{} SET {}=? WHERE address=? AND port=? AND client_type=?;",
db::system_keyspace::CLIENTS, column_literal);
return db::qctx->execute_cql(req, std::forward(value), std::move(addr), port, to_string(ct)).discard_result();
}
};