Instead of lengthy blurbs, switch to single-line, machine-readable standardized (https://spdx.dev) license identifiers. The Linux kernel switched long ago, so there is strong precedent. Three cases are handled: AGPL-only, Apache-only, and dual licensed. For the latter case, I chose (AGPL-3.0-or-later and Apache-2.0), reasoning that our changes are extensive enough to apply our license. The changes we applied mechanically with a script, except to licenses/README.md. Closes #9937
1011 lines
43 KiB
C++
1011 lines
43 KiB
C++
/*
|
|
* Copyright (C) 2015-present ScyllaDB
|
|
*/
|
|
|
|
/*
|
|
* SPDX-License-Identifier: AGPL-3.0-or-later
|
|
*/
|
|
|
|
#include "column_family.hh"
|
|
#include "api/api-doc/column_family.json.hh"
|
|
#include <vector>
|
|
#include <seastar/http/exception.hh>
|
|
#include "sstables/sstables.hh"
|
|
#include "sstables/metadata_collector.hh"
|
|
#include "utils/estimated_histogram.hh"
|
|
#include <algorithm>
|
|
#include "db/system_keyspace_view_types.hh"
|
|
#include "db/data_listeners.hh"
|
|
#include "storage_service.hh"
|
|
#include "unimplemented.hh"
|
|
|
|
extern logging::logger apilog;
|
|
|
|
namespace api {
|
|
using namespace httpd;
|
|
|
|
using namespace std;
|
|
using namespace json;
|
|
namespace cf = httpd::column_family_json;
|
|
|
|
std::tuple<sstring, sstring> parse_fully_qualified_cf_name(sstring name) {
|
|
auto pos = name.find("%3A");
|
|
size_t end;
|
|
if (pos == sstring::npos) {
|
|
pos = name.find(":");
|
|
if (pos == sstring::npos) {
|
|
throw bad_param_exception("Column family name should be in keyspace:column_family format");
|
|
}
|
|
end = pos + 1;
|
|
} else {
|
|
end = pos + 3;
|
|
}
|
|
return std::make_tuple(name.substr(0, pos), name.substr(end));
|
|
}
|
|
|
|
const utils::UUID& get_uuid(const sstring& ks, const sstring& cf, const replica::database& db) {
|
|
try {
|
|
return db.find_uuid(ks, cf);
|
|
} catch (replica::no_such_column_family& e) {
|
|
throw bad_param_exception(e.what());
|
|
}
|
|
}
|
|
|
|
const utils::UUID& get_uuid(const sstring& name, const replica::database& db) {
|
|
auto [ks, cf] = parse_fully_qualified_cf_name(name);
|
|
return get_uuid(ks, cf, db);
|
|
}
|
|
|
|
future<> foreach_column_family(http_context& ctx, const sstring& name, function<void(replica::column_family&)> f) {
|
|
auto uuid = get_uuid(name, ctx.db.local());
|
|
|
|
return ctx.db.invoke_on_all([f, uuid](replica::database& db) {
|
|
f(db.find_column_family(uuid));
|
|
});
|
|
}
|
|
|
|
future<json::json_return_type> get_cf_stats(http_context& ctx, const sstring& name,
|
|
int64_t replica::column_family_stats::*f) {
|
|
return map_reduce_cf(ctx, name, int64_t(0), [f](const replica::column_family& cf) {
|
|
return cf.get_stats().*f;
|
|
}, std::plus<int64_t>());
|
|
}
|
|
|
|
future<json::json_return_type> get_cf_stats(http_context& ctx,
|
|
int64_t replica::column_family_stats::*f) {
|
|
return map_reduce_cf(ctx, int64_t(0), [f](const replica::column_family& cf) {
|
|
return cf.get_stats().*f;
|
|
}, std::plus<int64_t>());
|
|
}
|
|
|
|
static future<json::json_return_type> get_cf_stats_count(http_context& ctx, const sstring& name,
|
|
utils::timed_rate_moving_average_and_histogram replica::column_family_stats::*f) {
|
|
return map_reduce_cf(ctx, name, int64_t(0), [f](const replica::column_family& cf) {
|
|
return (cf.get_stats().*f).hist.count;
|
|
}, std::plus<int64_t>());
|
|
}
|
|
|
|
static future<json::json_return_type> get_cf_stats_sum(http_context& ctx, const sstring& name,
|
|
utils::timed_rate_moving_average_and_histogram replica::column_family_stats::*f) {
|
|
auto uuid = get_uuid(name, ctx.db.local());
|
|
return ctx.db.map_reduce0([uuid, f](replica::database& db) {
|
|
// Histograms information is sample of the actual load
|
|
// so to get an estimation of sum, we multiply the mean
|
|
// with count. The information is gather in nano second,
|
|
// but reported in micro
|
|
replica::column_family& cf = db.find_column_family(uuid);
|
|
return ((cf.get_stats().*f).hist.count/1000.0) * (cf.get_stats().*f).hist.mean;
|
|
}, 0.0, std::plus<double>()).then([](double res) {
|
|
return make_ready_future<json::json_return_type>((int64_t)res);
|
|
});
|
|
}
|
|
|
|
|
|
static future<json::json_return_type> get_cf_stats_count(http_context& ctx,
|
|
utils::timed_rate_moving_average_and_histogram replica::column_family_stats::*f) {
|
|
return map_reduce_cf(ctx, int64_t(0), [f](const replica::column_family& cf) {
|
|
return (cf.get_stats().*f).hist.count;
|
|
}, std::plus<int64_t>());
|
|
}
|
|
|
|
static future<json::json_return_type> get_cf_histogram(http_context& ctx, const sstring& name,
|
|
utils::timed_rate_moving_average_and_histogram replica::column_family_stats::*f) {
|
|
utils::UUID uuid = get_uuid(name, ctx.db.local());
|
|
return ctx.db.map_reduce0([f, uuid](const replica::database& p) {
|
|
return (p.find_column_family(uuid).get_stats().*f).hist;},
|
|
utils::ihistogram(),
|
|
std::plus<utils::ihistogram>())
|
|
.then([](const utils::ihistogram& val) {
|
|
return make_ready_future<json::json_return_type>(to_json(val));
|
|
});
|
|
}
|
|
|
|
static future<json::json_return_type> get_cf_histogram(http_context& ctx, utils::timed_rate_moving_average_and_histogram replica::column_family_stats::*f) {
|
|
std::function<utils::ihistogram(const replica::database&)> fun = [f] (const replica::database& db) {
|
|
utils::ihistogram res;
|
|
for (auto i : db.get_column_families()) {
|
|
res += (i.second->get_stats().*f).hist;
|
|
}
|
|
return res;
|
|
};
|
|
return ctx.db.map(fun).then([](const std::vector<utils::ihistogram> &res) {
|
|
std::vector<httpd::utils_json::histogram> r;
|
|
boost::copy(res | boost::adaptors::transformed(to_json), std::back_inserter(r));
|
|
return make_ready_future<json::json_return_type>(r);
|
|
});
|
|
}
|
|
|
|
static future<json::json_return_type> get_cf_rate_and_histogram(http_context& ctx, const sstring& name,
|
|
utils::timed_rate_moving_average_and_histogram replica::column_family_stats::*f) {
|
|
utils::UUID uuid = get_uuid(name, ctx.db.local());
|
|
return ctx.db.map_reduce0([f, uuid](const replica::database& p) {
|
|
return (p.find_column_family(uuid).get_stats().*f).rate();},
|
|
utils::rate_moving_average_and_histogram(),
|
|
std::plus<utils::rate_moving_average_and_histogram>())
|
|
.then([](const utils::rate_moving_average_and_histogram& val) {
|
|
return make_ready_future<json::json_return_type>(timer_to_json(val));
|
|
});
|
|
}
|
|
|
|
static future<json::json_return_type> get_cf_rate_and_histogram(http_context& ctx, utils::timed_rate_moving_average_and_histogram replica::column_family_stats::*f) {
|
|
std::function<utils::rate_moving_average_and_histogram(const replica::database&)> fun = [f] (const replica::database& db) {
|
|
utils::rate_moving_average_and_histogram res;
|
|
for (auto i : db.get_column_families()) {
|
|
res += (i.second->get_stats().*f).rate();
|
|
}
|
|
return res;
|
|
};
|
|
return ctx.db.map(fun).then([](const std::vector<utils::rate_moving_average_and_histogram> &res) {
|
|
std::vector<httpd::utils_json::rate_moving_average_and_histogram> r;
|
|
boost::copy(res | boost::adaptors::transformed(timer_to_json), std::back_inserter(r));
|
|
return make_ready_future<json::json_return_type>(r);
|
|
});
|
|
}
|
|
|
|
static future<json::json_return_type> get_cf_unleveled_sstables(http_context& ctx, const sstring& name) {
|
|
return map_reduce_cf(ctx, name, int64_t(0), [](const replica::column_family& cf) {
|
|
return cf.get_unleveled_sstables();
|
|
}, std::plus<int64_t>());
|
|
}
|
|
|
|
static int64_t min_partition_size(replica::column_family& cf) {
|
|
int64_t res = INT64_MAX;
|
|
for (auto sstables = cf.get_sstables(); auto& i : *sstables) {
|
|
res = std::min(res, i->get_stats_metadata().estimated_partition_size.min());
|
|
}
|
|
return (res == INT64_MAX) ? 0 : res;
|
|
}
|
|
|
|
static int64_t max_partition_size(replica::column_family& cf) {
|
|
int64_t res = 0;
|
|
for (auto sstables = cf.get_sstables(); auto& i : *sstables) {
|
|
res = std::max(i->get_stats_metadata().estimated_partition_size.max(), res);
|
|
}
|
|
return res;
|
|
}
|
|
|
|
static integral_ratio_holder mean_partition_size(replica::column_family& cf) {
|
|
integral_ratio_holder res;
|
|
for (auto sstables = cf.get_sstables(); auto& i : *sstables) {
|
|
auto c = i->get_stats_metadata().estimated_partition_size.count();
|
|
res.sub += i->get_stats_metadata().estimated_partition_size.mean() * c;
|
|
res.total += c;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
static std::unordered_map<sstring, uint64_t> merge_maps(std::unordered_map<sstring, uint64_t> a,
|
|
const std::unordered_map<sstring, uint64_t>& b) {
|
|
a.insert(b.begin(), b.end());
|
|
return a;
|
|
}
|
|
|
|
static json::json_return_type sum_map(const std::unordered_map<sstring, uint64_t>& val) {
|
|
uint64_t res = 0;
|
|
for (auto i : val) {
|
|
res += i.second;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
static future<json::json_return_type> sum_sstable(http_context& ctx, const sstring name, bool total) {
|
|
auto uuid = get_uuid(name, ctx.db.local());
|
|
return ctx.db.map_reduce0([uuid, total](replica::database& db) {
|
|
std::unordered_map<sstring, uint64_t> m;
|
|
auto sstables = (total) ? db.find_column_family(uuid).get_sstables_including_compacted_undeleted() :
|
|
db.find_column_family(uuid).get_sstables();
|
|
for (auto t : *sstables) {
|
|
m[t->get_filename()] = t->bytes_on_disk();
|
|
}
|
|
return m;
|
|
}, std::unordered_map<sstring, uint64_t>(), merge_maps).
|
|
then([](const std::unordered_map<sstring, uint64_t>& val) {
|
|
return sum_map(val);
|
|
});
|
|
}
|
|
|
|
|
|
static future<json::json_return_type> sum_sstable(http_context& ctx, bool total) {
|
|
return map_reduce_cf_raw(ctx, std::unordered_map<sstring, uint64_t>(), [total](replica::column_family& cf) {
|
|
std::unordered_map<sstring, uint64_t> m;
|
|
auto sstables = (total) ? cf.get_sstables_including_compacted_undeleted() :
|
|
cf.get_sstables();
|
|
for (auto t : *sstables) {
|
|
m[t->get_filename()] = t->bytes_on_disk();
|
|
}
|
|
return m;
|
|
},merge_maps).then([](const std::unordered_map<sstring, uint64_t>& val) {
|
|
return sum_map(val);
|
|
});
|
|
}
|
|
|
|
future<json::json_return_type> map_reduce_cf_time_histogram(http_context& ctx, const sstring& name, std::function<utils::time_estimated_histogram(const replica::column_family&)> f) {
|
|
return map_reduce_cf_raw(ctx, name, utils::time_estimated_histogram(), f, utils::time_estimated_histogram_merge).then([](const utils::time_estimated_histogram& res) {
|
|
return make_ready_future<json::json_return_type>(time_to_json_histogram(res));
|
|
});
|
|
}
|
|
|
|
template <typename T>
|
|
class sum_ratio {
|
|
uint64_t _n = 0;
|
|
T _total = 0;
|
|
public:
|
|
void operator()(T value) {
|
|
if (value > 0) {
|
|
_total += value;
|
|
_n++;
|
|
}
|
|
}
|
|
// Returns average value of all registered ratios.
|
|
T get() && {
|
|
return _n ? (_total / _n) : T(0);
|
|
}
|
|
};
|
|
|
|
static double get_compression_ratio(replica::column_family& cf) {
|
|
sum_ratio<double> result;
|
|
for (auto sstables = cf.get_sstables(); auto& i : *sstables) {
|
|
auto compression_ratio = i->get_compression_ratio();
|
|
if (compression_ratio != sstables::metadata_collector::NO_COMPRESSION_RATIO) {
|
|
result(compression_ratio);
|
|
}
|
|
}
|
|
return std::move(result).get();
|
|
}
|
|
|
|
static std::vector<uint64_t> concat_sstable_count_per_level(std::vector<uint64_t> a, std::vector<uint64_t>&& b) {
|
|
a.resize(std::max(a.size(), b.size()), 0UL);
|
|
for (auto i = 0U; i < b.size(); i++) {
|
|
a[i] += b[i];
|
|
}
|
|
return a;
|
|
}
|
|
|
|
ratio_holder filter_false_positive_as_ratio_holder(const sstables::shared_sstable& sst) {
|
|
double f = sst->filter_get_false_positive();
|
|
return ratio_holder(f + sst->filter_get_true_positive(), f);
|
|
}
|
|
|
|
ratio_holder filter_recent_false_positive_as_ratio_holder(const sstables::shared_sstable& sst) {
|
|
double f = sst->filter_get_recent_false_positive();
|
|
return ratio_holder(f + sst->filter_get_recent_true_positive(), f);
|
|
}
|
|
|
|
void set_column_family(http_context& ctx, routes& r) {
|
|
cf::get_column_family_name.set(r, [&ctx] (const_req req){
|
|
vector<sstring> res;
|
|
for (auto i: ctx.db.local().get_column_families_mapping()) {
|
|
res.push_back(i.first.first + ":" + i.first.second);
|
|
}
|
|
return res;
|
|
});
|
|
|
|
cf::get_column_family.set(r, [&ctx] (std::unique_ptr<request> req){
|
|
std::list<cf::column_family_info> res;
|
|
for (auto i: ctx.db.local().get_column_families_mapping()) {
|
|
cf::column_family_info info;
|
|
info.ks = i.first.first;
|
|
info.cf = i.first.second;
|
|
info.type = "ColumnFamilies";
|
|
res.push_back(info);
|
|
}
|
|
return make_ready_future<json::json_return_type>(json::stream_range_as_array(std::move(res), std::identity()));
|
|
});
|
|
|
|
cf::get_column_family_name_keyspace.set(r, [&ctx] (const_req req){
|
|
vector<sstring> res;
|
|
for (auto i = ctx.db.local().get_keyspaces().cbegin(); i!= ctx.db.local().get_keyspaces().cend(); i++) {
|
|
res.push_back(i->first);
|
|
}
|
|
return res;
|
|
});
|
|
|
|
cf::get_memtable_columns_count.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], uint64_t{0}, [](replica::column_family& cf) {
|
|
return cf.active_memtable().partition_count();
|
|
}, std::plus<>());
|
|
});
|
|
|
|
cf::get_all_memtable_columns_count.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, uint64_t{0}, [](replica::column_family& cf) {
|
|
return cf.active_memtable().partition_count();
|
|
}, std::plus<>());
|
|
});
|
|
|
|
cf::get_memtable_on_heap_size.set(r, [] (const_req req) {
|
|
return 0;
|
|
});
|
|
|
|
cf::get_all_memtable_on_heap_size.set(r, [] (const_req req) {
|
|
return 0;
|
|
});
|
|
|
|
cf::get_memtable_off_heap_size.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], int64_t(0), [](replica::column_family& cf) {
|
|
return cf.active_memtable().region().occupancy().total_space();
|
|
}, std::plus<int64_t>());
|
|
});
|
|
|
|
cf::get_all_memtable_off_heap_size.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, int64_t(0), [](replica::column_family& cf) {
|
|
return cf.active_memtable().region().occupancy().total_space();
|
|
}, std::plus<int64_t>());
|
|
});
|
|
|
|
cf::get_memtable_live_data_size.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], int64_t(0), [](replica::column_family& cf) {
|
|
return cf.active_memtable().region().occupancy().used_space();
|
|
}, std::plus<int64_t>());
|
|
});
|
|
|
|
cf::get_all_memtable_live_data_size.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, int64_t(0), [](replica::column_family& cf) {
|
|
return cf.active_memtable().region().occupancy().used_space();
|
|
}, std::plus<int64_t>());
|
|
});
|
|
|
|
cf::get_cf_all_memtables_on_heap_size.set(r, [] (const_req req) {
|
|
return 0;
|
|
});
|
|
|
|
cf::get_all_cf_all_memtables_on_heap_size.set(r, [] (const_req req) {
|
|
return 0;
|
|
});
|
|
|
|
cf::get_cf_all_memtables_off_heap_size.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
warn(unimplemented::cause::INDEXES);
|
|
return map_reduce_cf(ctx, req->param["name"], int64_t(0), [](replica::column_family& cf) {
|
|
return cf.occupancy().total_space();
|
|
}, std::plus<int64_t>());
|
|
});
|
|
|
|
cf::get_all_cf_all_memtables_off_heap_size.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
warn(unimplemented::cause::INDEXES);
|
|
return ctx.db.map_reduce0([](const replica::database& db){
|
|
return db.dirty_memory_region_group().memory_used();
|
|
}, int64_t(0), std::plus<int64_t>()).then([](int res) {
|
|
return make_ready_future<json::json_return_type>(res);
|
|
});
|
|
});
|
|
|
|
cf::get_cf_all_memtables_live_data_size.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
warn(unimplemented::cause::INDEXES);
|
|
return map_reduce_cf(ctx, req->param["name"], int64_t(0), [](replica::column_family& cf) {
|
|
return cf.occupancy().used_space();
|
|
}, std::plus<int64_t>());
|
|
});
|
|
|
|
cf::get_all_cf_all_memtables_live_data_size.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
warn(unimplemented::cause::INDEXES);
|
|
return map_reduce_cf(ctx, int64_t(0), [](replica::column_family& cf) {
|
|
return cf.active_memtable().region().occupancy().used_space();
|
|
}, std::plus<int64_t>());
|
|
});
|
|
|
|
cf::get_memtable_switch_count.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_stats(ctx,req->param["name"] ,&replica::column_family_stats::memtable_switch_count);
|
|
});
|
|
|
|
cf::get_all_memtable_switch_count.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_stats(ctx, &replica::column_family_stats::memtable_switch_count);
|
|
});
|
|
|
|
// FIXME: this refers to partitions, not rows.
|
|
cf::get_estimated_row_size_histogram.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], utils::estimated_histogram(0), [](replica::column_family& cf) {
|
|
utils::estimated_histogram res(0);
|
|
for (auto sstables = cf.get_sstables(); auto& i : *sstables) {
|
|
res.merge(i->get_stats_metadata().estimated_partition_size);
|
|
}
|
|
return res;
|
|
},
|
|
utils::estimated_histogram_merge, utils_json::estimated_histogram());
|
|
});
|
|
|
|
// FIXME: this refers to partitions, not rows.
|
|
cf::get_estimated_row_count.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], int64_t(0), [](replica::column_family& cf) {
|
|
uint64_t res = 0;
|
|
for (auto sstables = cf.get_sstables(); auto& i : *sstables) {
|
|
res += i->get_stats_metadata().estimated_partition_size.count();
|
|
}
|
|
return res;
|
|
},
|
|
std::plus<uint64_t>());
|
|
});
|
|
|
|
cf::get_estimated_column_count_histogram.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], utils::estimated_histogram(0), [](replica::column_family& cf) {
|
|
utils::estimated_histogram res(0);
|
|
for (auto sstables = cf.get_sstables(); auto& i : *sstables) {
|
|
res.merge(i->get_stats_metadata().estimated_cells_count);
|
|
}
|
|
return res;
|
|
},
|
|
utils::estimated_histogram_merge, utils_json::estimated_histogram());
|
|
});
|
|
|
|
cf::get_all_compression_ratio.set(r, [] (std::unique_ptr<request> req) {
|
|
//TBD
|
|
unimplemented();
|
|
return make_ready_future<json::json_return_type>(0);
|
|
});
|
|
|
|
cf::get_pending_flushes.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_stats(ctx,req->param["name"] ,&replica::column_family_stats::pending_flushes);
|
|
});
|
|
|
|
cf::get_all_pending_flushes.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_stats(ctx, &replica::column_family_stats::pending_flushes);
|
|
});
|
|
|
|
cf::get_read.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_stats_count(ctx,req->param["name"] ,&replica::column_family_stats::reads);
|
|
});
|
|
|
|
cf::get_all_read.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_stats_count(ctx, &replica::column_family_stats::reads);
|
|
});
|
|
|
|
cf::get_write.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_stats_count(ctx, req->param["name"] ,&replica::column_family_stats::writes);
|
|
});
|
|
|
|
cf::get_all_write.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_stats_count(ctx, &replica::column_family_stats::writes);
|
|
});
|
|
|
|
cf::get_read_latency_histogram_depricated.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_histogram(ctx, req->param["name"], &replica::column_family_stats::reads);
|
|
});
|
|
|
|
cf::get_read_latency_histogram.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_rate_and_histogram(ctx, req->param["name"], &replica::column_family_stats::reads);
|
|
});
|
|
|
|
cf::get_read_latency.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_stats_sum(ctx,req->param["name"] ,&replica::column_family_stats::reads);
|
|
});
|
|
|
|
cf::get_write_latency.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_stats_sum(ctx, req->param["name"] ,&replica::column_family_stats::writes);
|
|
});
|
|
|
|
cf::get_all_read_latency_histogram_depricated.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_histogram(ctx, &replica::column_family_stats::writes);
|
|
});
|
|
|
|
cf::get_all_read_latency_histogram.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_rate_and_histogram(ctx, &replica::column_family_stats::writes);
|
|
});
|
|
|
|
cf::get_write_latency_histogram_depricated.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_histogram(ctx, req->param["name"], &replica::column_family_stats::writes);
|
|
});
|
|
|
|
cf::get_write_latency_histogram.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_rate_and_histogram(ctx, req->param["name"], &replica::column_family_stats::writes);
|
|
});
|
|
|
|
cf::get_all_write_latency_histogram_depricated.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_histogram(ctx, &replica::column_family_stats::writes);
|
|
});
|
|
|
|
cf::get_all_write_latency_histogram.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_rate_and_histogram(ctx, &replica::column_family_stats::writes);
|
|
});
|
|
|
|
cf::get_pending_compactions.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], int64_t(0), [](replica::column_family& cf) {
|
|
return cf.get_compaction_strategy().estimated_pending_compactions(cf.as_table_state());
|
|
}, std::plus<int64_t>());
|
|
});
|
|
|
|
cf::get_all_pending_compactions.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, int64_t(0), [](replica::column_family& cf) {
|
|
return cf.get_compaction_strategy().estimated_pending_compactions(cf.as_table_state());
|
|
}, std::plus<int64_t>());
|
|
});
|
|
|
|
cf::get_live_ss_table_count.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_stats(ctx, req->param["name"], &replica::column_family_stats::live_sstable_count);
|
|
});
|
|
|
|
cf::get_all_live_ss_table_count.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_stats(ctx, &replica::column_family_stats::live_sstable_count);
|
|
});
|
|
|
|
cf::get_unleveled_sstables.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_unleveled_sstables(ctx, req->param["name"]);
|
|
});
|
|
|
|
cf::get_live_disk_space_used.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return sum_sstable(ctx, req->param["name"], false);
|
|
});
|
|
|
|
cf::get_all_live_disk_space_used.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return sum_sstable(ctx, false);
|
|
});
|
|
|
|
cf::get_total_disk_space_used.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return sum_sstable(ctx, req->param["name"], true);
|
|
});
|
|
|
|
cf::get_all_total_disk_space_used.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return sum_sstable(ctx, true);
|
|
});
|
|
|
|
// FIXME: this refers to partitions, not rows.
|
|
cf::get_min_row_size.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], INT64_MAX, min_partition_size, min_int64);
|
|
});
|
|
|
|
// FIXME: this refers to partitions, not rows.
|
|
cf::get_all_min_row_size.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, INT64_MAX, min_partition_size, min_int64);
|
|
});
|
|
|
|
// FIXME: this refers to partitions, not rows.
|
|
cf::get_max_row_size.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], int64_t(0), max_partition_size, max_int64);
|
|
});
|
|
|
|
// FIXME: this refers to partitions, not rows.
|
|
cf::get_all_max_row_size.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, int64_t(0), max_partition_size, max_int64);
|
|
});
|
|
|
|
// FIXME: this refers to partitions, not rows.
|
|
cf::get_mean_row_size.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
// Cassandra 3.x mean values are truncated as integrals.
|
|
return map_reduce_cf(ctx, req->param["name"], integral_ratio_holder(), mean_partition_size, std::plus<integral_ratio_holder>());
|
|
});
|
|
|
|
// FIXME: this refers to partitions, not rows.
|
|
cf::get_all_mean_row_size.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
// Cassandra 3.x mean values are truncated as integrals.
|
|
return map_reduce_cf(ctx, integral_ratio_holder(), mean_partition_size, std::plus<integral_ratio_holder>());
|
|
});
|
|
|
|
cf::get_bloom_filter_false_positives.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], uint64_t(0), [] (replica::column_family& cf) {
|
|
auto sstables = cf.get_sstables();
|
|
return std::accumulate(sstables->begin(), sstables->end(), uint64_t(0), [](uint64_t s, auto& sst) {
|
|
return s + sst->filter_get_false_positive();
|
|
});
|
|
}, std::plus<uint64_t>());
|
|
});
|
|
|
|
cf::get_all_bloom_filter_false_positives.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, uint64_t(0), [] (replica::column_family& cf) {
|
|
auto sstables = cf.get_sstables();
|
|
return std::accumulate(sstables->begin(), sstables->end(), uint64_t(0), [](uint64_t s, auto& sst) {
|
|
return s + sst->filter_get_false_positive();
|
|
});
|
|
}, std::plus<uint64_t>());
|
|
});
|
|
|
|
cf::get_recent_bloom_filter_false_positives.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], uint64_t(0), [] (replica::column_family& cf) {
|
|
auto sstables = cf.get_sstables();
|
|
return std::accumulate(sstables->begin(), sstables->end(), uint64_t(0), [](uint64_t s, auto& sst) {
|
|
return s + sst->filter_get_recent_false_positive();
|
|
});
|
|
}, std::plus<uint64_t>());
|
|
});
|
|
|
|
cf::get_all_recent_bloom_filter_false_positives.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, uint64_t(0), [] (replica::column_family& cf) {
|
|
auto sstables = cf.get_sstables();
|
|
return std::accumulate(sstables->begin(), sstables->end(), uint64_t(0), [](uint64_t s, auto& sst) {
|
|
return s + sst->filter_get_recent_false_positive();
|
|
});
|
|
}, std::plus<uint64_t>());
|
|
});
|
|
|
|
cf::get_bloom_filter_false_ratio.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], ratio_holder(), [] (replica::column_family& cf) {
|
|
return boost::accumulate(*cf.get_sstables() | boost::adaptors::transformed(filter_false_positive_as_ratio_holder), ratio_holder());
|
|
}, std::plus<>());
|
|
});
|
|
|
|
cf::get_all_bloom_filter_false_ratio.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, ratio_holder(), [] (replica::column_family& cf) {
|
|
return boost::accumulate(*cf.get_sstables() | boost::adaptors::transformed(filter_false_positive_as_ratio_holder), ratio_holder());
|
|
}, std::plus<>());
|
|
});
|
|
|
|
cf::get_recent_bloom_filter_false_ratio.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], ratio_holder(), [] (replica::column_family& cf) {
|
|
return boost::accumulate(*cf.get_sstables() | boost::adaptors::transformed(filter_recent_false_positive_as_ratio_holder), ratio_holder());
|
|
}, std::plus<>());
|
|
});
|
|
|
|
cf::get_all_recent_bloom_filter_false_ratio.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, ratio_holder(), [] (replica::column_family& cf) {
|
|
return boost::accumulate(*cf.get_sstables() | boost::adaptors::transformed(filter_recent_false_positive_as_ratio_holder), ratio_holder());
|
|
}, std::plus<>());
|
|
});
|
|
|
|
cf::get_bloom_filter_disk_space_used.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], uint64_t(0), [] (replica::column_family& cf) {
|
|
auto sstables = cf.get_sstables();
|
|
return std::accumulate(sstables->begin(), sstables->end(), uint64_t(0), [](uint64_t s, auto& sst) {
|
|
return s + sst->filter_size();
|
|
});
|
|
}, std::plus<uint64_t>());
|
|
});
|
|
|
|
cf::get_all_bloom_filter_disk_space_used.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, uint64_t(0), [] (replica::column_family& cf) {
|
|
auto sstables = cf.get_sstables();
|
|
return std::accumulate(sstables->begin(), sstables->end(), uint64_t(0), [](uint64_t s, auto& sst) {
|
|
return s + sst->filter_size();
|
|
});
|
|
}, std::plus<uint64_t>());
|
|
});
|
|
|
|
cf::get_bloom_filter_off_heap_memory_used.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], uint64_t(0), [] (replica::column_family& cf) {
|
|
auto sstables = cf.get_sstables();
|
|
return std::accumulate(sstables->begin(), sstables->end(), uint64_t(0), [](uint64_t s, auto& sst) {
|
|
return s + sst->filter_memory_size();
|
|
});
|
|
}, std::plus<uint64_t>());
|
|
});
|
|
|
|
cf::get_all_bloom_filter_off_heap_memory_used.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, uint64_t(0), [] (replica::column_family& cf) {
|
|
auto sstables = cf.get_sstables();
|
|
return std::accumulate(sstables->begin(), sstables->end(), uint64_t(0), [](uint64_t s, auto& sst) {
|
|
return s + sst->filter_memory_size();
|
|
});
|
|
}, std::plus<uint64_t>());
|
|
});
|
|
|
|
cf::get_index_summary_off_heap_memory_used.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], uint64_t(0), [] (replica::column_family& cf) {
|
|
auto sstables = cf.get_sstables();
|
|
return std::accumulate(sstables->begin(), sstables->end(), uint64_t(0), [](uint64_t s, auto& sst) {
|
|
return s + sst->get_summary().memory_footprint();
|
|
});
|
|
}, std::plus<uint64_t>());
|
|
});
|
|
|
|
cf::get_all_index_summary_off_heap_memory_used.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, uint64_t(0), [] (replica::column_family& cf) {
|
|
auto sstables = cf.get_sstables();
|
|
return std::accumulate(sstables->begin(), sstables->end(), uint64_t(0), [](uint64_t s, auto& sst) {
|
|
return s + sst->get_summary().memory_footprint();
|
|
});
|
|
}, std::plus<uint64_t>());
|
|
});
|
|
|
|
cf::get_compression_metadata_off_heap_memory_used.set(r, [] (std::unique_ptr<request> req) {
|
|
//TBD
|
|
// FIXME
|
|
// We are missing the off heap memory calculation
|
|
// Return 0 is the wrong value. It's a work around
|
|
// until the memory calculation will be available
|
|
//auto id = get_uuid(req->param["name"], ctx.db.local());
|
|
return make_ready_future<json::json_return_type>(0);
|
|
});
|
|
|
|
cf::get_all_compression_metadata_off_heap_memory_used.set(r, [] (std::unique_ptr<request> req) {
|
|
//TBD
|
|
unimplemented();
|
|
return make_ready_future<json::json_return_type>(0);
|
|
});
|
|
|
|
cf::get_speculative_retries.set(r, [] (std::unique_ptr<request> req) {
|
|
//TBD
|
|
unimplemented();
|
|
//auto id = get_uuid(req->param["name"], ctx.db.local());
|
|
return make_ready_future<json::json_return_type>(0);
|
|
});
|
|
|
|
cf::get_all_speculative_retries.set(r, [] (std::unique_ptr<request> req) {
|
|
//TBD
|
|
unimplemented();
|
|
return make_ready_future<json::json_return_type>(0);
|
|
});
|
|
|
|
cf::get_key_cache_hit_rate.set(r, [] (std::unique_ptr<request> req) {
|
|
//TBD
|
|
unimplemented();
|
|
//auto id = get_uuid(req->param["name"], ctx.db.local());
|
|
return make_ready_future<json::json_return_type>(0);
|
|
});
|
|
|
|
cf::get_true_snapshots_size.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
auto uuid = get_uuid(req->param["name"], ctx.db.local());
|
|
return ctx.db.local().find_column_family(uuid).get_snapshot_details().then([](
|
|
const std::unordered_map<sstring, replica::column_family::snapshot_details>& sd) {
|
|
int64_t res = 0;
|
|
for (auto i : sd) {
|
|
res += i.second.total;
|
|
}
|
|
return make_ready_future<json::json_return_type>(res);
|
|
});
|
|
});
|
|
|
|
cf::get_all_true_snapshots_size.set(r, [] (std::unique_ptr<request> req) {
|
|
//TBD
|
|
unimplemented();
|
|
return make_ready_future<json::json_return_type>(0);
|
|
});
|
|
|
|
cf::get_row_cache_hit_out_of_range.set(r, [] (std::unique_ptr<request> req) {
|
|
//TBD
|
|
unimplemented();
|
|
//auto id = get_uuid(req->param["name"], ctx.db.local());
|
|
return make_ready_future<json::json_return_type>(0);
|
|
});
|
|
|
|
cf::get_all_row_cache_hit_out_of_range.set(r, [] (std::unique_ptr<request> req) {
|
|
//TBD
|
|
unimplemented();
|
|
return make_ready_future<json::json_return_type>(0);
|
|
});
|
|
|
|
cf::get_row_cache_hit.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf_raw(ctx, req->param["name"], utils::rate_moving_average(), [](const replica::column_family& cf) {
|
|
return cf.get_row_cache().stats().hits.rate();
|
|
}, std::plus<utils::rate_moving_average>()).then([](const utils::rate_moving_average& m) {
|
|
return make_ready_future<json::json_return_type>(meter_to_json(m));
|
|
});
|
|
});
|
|
|
|
cf::get_all_row_cache_hit.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf_raw(ctx, utils::rate_moving_average(), [](const replica::column_family& cf) {
|
|
return cf.get_row_cache().stats().hits.rate();
|
|
}, std::plus<utils::rate_moving_average>()).then([](const utils::rate_moving_average& m) {
|
|
return make_ready_future<json::json_return_type>(meter_to_json(m));
|
|
});
|
|
});
|
|
|
|
cf::get_row_cache_miss.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf_raw(ctx, req->param["name"], utils::rate_moving_average(), [](const replica::column_family& cf) {
|
|
return cf.get_row_cache().stats().misses.rate();
|
|
}, std::plus<utils::rate_moving_average>()).then([](const utils::rate_moving_average& m) {
|
|
return make_ready_future<json::json_return_type>(meter_to_json(m));
|
|
});
|
|
});
|
|
|
|
cf::get_all_row_cache_miss.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf_raw(ctx, utils::rate_moving_average(), [](const replica::column_family& cf) {
|
|
return cf.get_row_cache().stats().misses.rate();
|
|
}, std::plus<utils::rate_moving_average>()).then([](const utils::rate_moving_average& m) {
|
|
return make_ready_future<json::json_return_type>(meter_to_json(m));
|
|
});
|
|
|
|
});
|
|
|
|
cf::get_cas_prepare.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf_time_histogram(ctx, req->param["name"], [](const replica::column_family& cf) {
|
|
return cf.get_stats().estimated_cas_prepare;
|
|
});
|
|
});
|
|
|
|
cf::get_cas_propose.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf_time_histogram(ctx, req->param["name"], [](const replica::column_family& cf) {
|
|
return cf.get_stats().estimated_cas_accept;
|
|
});
|
|
});
|
|
|
|
cf::get_cas_commit.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf_time_histogram(ctx, req->param["name"], [](const replica::column_family& cf) {
|
|
return cf.get_stats().estimated_cas_learn;
|
|
});
|
|
});
|
|
|
|
cf::get_sstables_per_read_histogram.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return map_reduce_cf(ctx, req->param["name"], utils::estimated_histogram(0), [](replica::column_family& cf) {
|
|
return cf.get_stats().estimated_sstable_per_read;
|
|
},
|
|
utils::estimated_histogram_merge, utils_json::estimated_histogram());
|
|
});
|
|
|
|
cf::get_tombstone_scanned_histogram.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_histogram(ctx, req->param["name"], &replica::column_family_stats::tombstone_scanned);
|
|
});
|
|
|
|
cf::get_live_scanned_histogram.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
return get_cf_histogram(ctx, req->param["name"], &replica::column_family_stats::live_scanned);
|
|
});
|
|
|
|
cf::get_col_update_time_delta_histogram.set(r, [] (std::unique_ptr<request> req) {
|
|
//TBD
|
|
unimplemented();
|
|
//auto id = get_uuid(req->param["name"], ctx.db.local());
|
|
std::vector<double> res;
|
|
return make_ready_future<json::json_return_type>(res);
|
|
});
|
|
|
|
cf::get_auto_compaction.set(r, [&ctx] (const_req req) {
|
|
const utils::UUID& uuid = get_uuid(req.param["name"], ctx.db.local());
|
|
replica::column_family& cf = ctx.db.local().find_column_family(uuid);
|
|
return !cf.is_auto_compaction_disabled_by_user();
|
|
});
|
|
|
|
cf::enable_auto_compaction.set(r, [&ctx](std::unique_ptr<request> req) {
|
|
return ctx.db.invoke_on(0, [&ctx, req = std::move(req)] (replica::database& db) {
|
|
auto g = replica::database::autocompaction_toggle_guard(db);
|
|
return foreach_column_family(ctx, req->param["name"], [](replica::column_family &cf) {
|
|
cf.enable_auto_compaction();
|
|
}).then([g = std::move(g)] {
|
|
return make_ready_future<json::json_return_type>(json_void());
|
|
});
|
|
});
|
|
});
|
|
|
|
cf::disable_auto_compaction.set(r, [&ctx](std::unique_ptr<request> req) {
|
|
return ctx.db.invoke_on(0, [&ctx, req = std::move(req)] (replica::database& db) {
|
|
auto g = replica::database::autocompaction_toggle_guard(db);
|
|
return foreach_column_family(ctx, req->param["name"], [](replica::column_family &cf) {
|
|
return cf.disable_auto_compaction();
|
|
}).then([g = std::move(g)] {
|
|
return make_ready_future<json::json_return_type>(json_void());
|
|
});
|
|
});
|
|
});
|
|
|
|
cf::get_built_indexes.set(r, [&ctx](std::unique_ptr<request> req) {
|
|
auto ks_cf = parse_fully_qualified_cf_name(req->param["name"]);
|
|
auto&& ks = std::get<0>(ks_cf);
|
|
auto&& cf_name = std::get<1>(ks_cf);
|
|
return db::system_keyspace::load_view_build_progress().then([ks, cf_name, &ctx](const std::vector<db::system_keyspace_view_build_progress>& vb) mutable {
|
|
std::set<sstring> vp;
|
|
for (auto b : vb) {
|
|
if (b.view.first == ks) {
|
|
vp.insert(b.view.second);
|
|
}
|
|
}
|
|
std::vector<sstring> res;
|
|
auto uuid = get_uuid(ks, cf_name, ctx.db.local());
|
|
replica::column_family& cf = ctx.db.local().find_column_family(uuid);
|
|
res.reserve(cf.get_index_manager().list_indexes().size());
|
|
for (auto&& i : cf.get_index_manager().list_indexes()) {
|
|
if (!vp.contains(secondary_index::index_table_name(i.metadata().name()))) {
|
|
res.emplace_back(i.metadata().name());
|
|
}
|
|
}
|
|
return make_ready_future<json::json_return_type>(res);
|
|
});
|
|
});
|
|
|
|
cf::get_compression_metadata_off_heap_memory_used.set(r, [](const_req) {
|
|
// FIXME
|
|
// Currently there are no information on the compression
|
|
// metadata, so we return 0
|
|
return 0;
|
|
});
|
|
|
|
cf::get_compression_parameters.set(r, [](const_req) {
|
|
// FIXME
|
|
// Currently there are no compression parameters available
|
|
// so we return an empty map
|
|
return std::vector<sstring>();
|
|
});
|
|
|
|
cf::get_compression_ratio.set(r, [&ctx](std::unique_ptr<request> req) {
|
|
auto uuid = get_uuid(req->param["name"], ctx.db.local());
|
|
|
|
return ctx.db.map_reduce(sum_ratio<double>(), [uuid](replica::database& db) {
|
|
replica::column_family& cf = db.find_column_family(uuid);
|
|
return make_ready_future<double>(get_compression_ratio(cf));
|
|
}).then([] (const double& result) {
|
|
return make_ready_future<json::json_return_type>(result);
|
|
});
|
|
});
|
|
|
|
cf::get_read_latency_estimated_histogram.set(r, [&ctx](std::unique_ptr<request> req) {
|
|
return map_reduce_cf_time_histogram(ctx, req->param["name"], [](const replica::column_family& cf) {
|
|
return cf.get_stats().estimated_read;
|
|
});
|
|
});
|
|
|
|
cf::get_write_latency_estimated_histogram.set(r, [&ctx](std::unique_ptr<request> req) {
|
|
return map_reduce_cf_time_histogram(ctx, req->param["name"], [](const replica::column_family& cf) {
|
|
return cf.get_stats().estimated_write;
|
|
});
|
|
});
|
|
|
|
cf::set_compaction_strategy_class.set(r, [&ctx](std::unique_ptr<request> req) {
|
|
sstring strategy = req->get_query_param("class_name");
|
|
return foreach_column_family(ctx, req->param["name"], [strategy](replica::column_family& cf) {
|
|
cf.set_compaction_strategy(sstables::compaction_strategy::type(strategy));
|
|
}).then([] {
|
|
return make_ready_future<json::json_return_type>(json_void());
|
|
});
|
|
});
|
|
|
|
cf::get_compaction_strategy_class.set(r, [&ctx](const_req req) {
|
|
return ctx.db.local().find_column_family(get_uuid(req.param["name"], ctx.db.local())).get_compaction_strategy().name();
|
|
});
|
|
|
|
cf::set_compression_parameters.set(r, [&ctx](std::unique_ptr<request> req) {
|
|
// TBD
|
|
unimplemented();
|
|
return make_ready_future<json::json_return_type>(json_void());
|
|
});
|
|
|
|
cf::set_crc_check_chance.set(r, [&ctx](std::unique_ptr<request> req) {
|
|
// TBD
|
|
unimplemented();
|
|
return make_ready_future<json::json_return_type>(json_void());
|
|
});
|
|
|
|
cf::get_sstable_count_per_level.set(r, [&ctx](std::unique_ptr<request> req) {
|
|
return map_reduce_cf_raw(ctx, req->param["name"], std::vector<uint64_t>(), [](const replica::column_family& cf) {
|
|
return cf.sstable_count_per_level();
|
|
}, concat_sstable_count_per_level).then([](const std::vector<uint64_t>& res) {
|
|
return make_ready_future<json::json_return_type>(res);
|
|
});
|
|
});
|
|
|
|
cf::get_sstables_for_key.set(r, [&ctx](std::unique_ptr<request> req) {
|
|
auto key = req->get_query_param("key");
|
|
auto uuid = get_uuid(req->param["name"], ctx.db.local());
|
|
|
|
return ctx.db.map_reduce0([key, uuid] (replica::database& db) {
|
|
return db.find_column_family(uuid).get_sstables_by_partition_key(key);
|
|
}, std::unordered_set<sstring>(),
|
|
[](std::unordered_set<sstring> a, std::unordered_set<sstring>&& b) mutable {
|
|
a.insert(b.begin(),b.end());
|
|
return a;
|
|
}).then([](const std::unordered_set<sstring>& res) {
|
|
return make_ready_future<json::json_return_type>(container_to_vec(res));
|
|
});
|
|
});
|
|
|
|
|
|
cf::toppartitions.set(r, [&ctx] (std::unique_ptr<request> req) {
|
|
auto name = req->param["name"];
|
|
auto [ks, cf] = parse_fully_qualified_cf_name(name);
|
|
|
|
api::req_param<std::chrono::milliseconds, unsigned> duration{*req, "duration", 1000ms};
|
|
api::req_param<unsigned> capacity(*req, "capacity", 256);
|
|
api::req_param<unsigned> list_size(*req, "list_size", 10);
|
|
|
|
apilog.info("toppartitions query: name={} duration={} list_size={} capacity={}",
|
|
name, duration.param, list_size.param, capacity.param);
|
|
|
|
return seastar::do_with(db::toppartitions_query(ctx.db, {{ks, cf}}, {}, duration.value, list_size, capacity), [&ctx] (db::toppartitions_query& q) {
|
|
return run_toppartitions_query(q, ctx, true);
|
|
});
|
|
});
|
|
|
|
cf::force_major_compaction.set(r, [&ctx](std::unique_ptr<request> req) {
|
|
if (req->get_query_param("split_output") != "") {
|
|
fail(unimplemented::cause::API);
|
|
}
|
|
return foreach_column_family(ctx, req->param["name"], [](replica::column_family &cf) {
|
|
return cf.compact_all_sstables();
|
|
}).then([] {
|
|
return make_ready_future<json::json_return_type>(json_void());
|
|
});
|
|
});
|
|
}
|
|
}
|