Files
scylladb/test/boost/extensions_test.cc
Piotr Dulikowski 6895b0e395 db::extensions: add shorthands for add_schema_extension
This abstract away a pattern used everywhere when adding a schema
extension.
2020-03-05 16:09:44 +01:00

138 lines
5.4 KiB
C++

/*
* Copyright (C) 2018 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 <http://www.gnu.org/licenses/>.
*/
#include <variant>
#include <stdexcept>
#include <seastar/core/future-util.hh>
#include <seastar/core/sleep.hh>
#include <seastar/testing/test_case.hh>
#include "test/lib/cql_test_env.hh"
#include "test/lib/cql_assertions.hh"
#include "db/extensions.hh"
#include "db/config.hh"
#include "serializer.hh"
#include "serializer_impl.hh"
#include "sstables/sstables.hh"
#include "cdc/cdc_extension.hh"
SEASTAR_TEST_CASE(simple_schema_extension) {
class dummy_ext : public schema_extension {
public:
dummy_ext(bytes b) : _bytes(b) {}
dummy_ext(const std::map<sstring, sstring>& map) : _bytes(ser::serialize_to_buffer<bytes>(map)) {}
dummy_ext(const sstring&) {
throw std::runtime_error("should not reach");
}
bytes serialize() const override {
return _bytes;
}
private:
bytes _bytes;
};
auto ext = std::make_shared<db::extensions>();
ext->add_schema_extension<dummy_ext>("los_lobos");
/**
* Ensure we can create a table with the extension, and that it (and its data) is preserved
* by the schema creation (which serializes and deserializes mutations)
*/
return do_with_cql_env([] (cql_test_env& e) {
return e.execute_cql("create table cf (id int primary key, value int) with los_lobos = { 'king of' : 'swing', 'ninja' : 'mission' };").discard_result().then([&e] {
auto& db = e.local_db();
auto& cf = db.find_column_family("ks", "cf");
auto s = cf.schema();
auto& ext = s->extensions().at("los_lobos");
BOOST_REQUIRE(!ext->is_placeholder());
BOOST_CHECK_EQUAL(ext->serialize(), dummy_ext(std::map<sstring, sstring>{{"king of", "swing"},{"ninja", "mission"}}).serialize());
});
}, ::make_shared<db::config>(ext));
}
using namespace sstables;
SEASTAR_TEST_CASE(simple_sstable_extension) {
static size_t counter = 0;
class dummy_ext : public file_io_extension {
public:
future<file> wrap_file(sstable& t, component_type type, file, open_flags flags) override {
if (type == component_type::Data && t.get_schema()->cf_name() == "cf") {
++counter;
}
return make_ready_future<file>();
}
};
auto ext = std::make_shared<db::extensions>();
ext->add_sstable_file_io_extension("los_lobos", std::make_unique<dummy_ext>());
counter = 0;
/**
* Ensure the extension is invoked on read/write of sstables, esp. data.
*/
return do_with_cql_env([] (cql_test_env& e) {
return e.execute_cql("create table cf (id int primary key, value int);").discard_result().then([&e] {
BOOST_REQUIRE(counter == 0);
// minimal data
return e.execute_cql("insert into ks.cf (id, value) values (1, 100);").discard_result().then([&e] {
// flush all shards
return e.db().invoke_on_all([](database& db) {
auto& cf = db.find_column_family("ks", "cf");
return cf.flush();
}).then([] {
BOOST_REQUIRE(counter > 1);
});
});
});
}, ::make_shared<db::config>(ext));
}
SEASTAR_TEST_CASE(cdc_schema_extension) {
auto ext = std::make_shared<db::extensions>();
// Extensions have to be registered here - config needs to have them before construction of test env.
ext->add_schema_extension<cdc::cdc_extension>(cdc::cdc_extension::NAME);
auto cfg = ::make_shared<db::config>(ext);
cfg->experimental_features({db::experimental_features_t::feature::CDC});
return do_with_cql_env([] (cql_test_env& e) {
auto assert_ext_correctness = [] (cql_test_env& e, cdc::cdc_extension expected_ext) {
auto& actual_ext = e.local_db().find_column_family("ks", "cf").schema()->extensions().at("cdc");
BOOST_REQUIRE(!actual_ext->is_placeholder());
BOOST_CHECK_EQUAL(actual_ext->serialize(), expected_ext.serialize());
};
return e.execute_cql("CREATE TABLE cf (id int PRIMARY KEY, value int) WITH cdc = {'enabled':'true','postimage':'true','preimage':'true','ttl':'6789'};")
.discard_result().then([&] {
assert_ext_correctness(e, cdc::cdc_extension{{{"enabled","true"},{"postimage","true"},{"preimage","true"},{"ttl","6789"}}});
}).then([&] {
return e.execute_cql("ALTER TABLE cf WITH cdc = {'enabled':'true','postimage':'false','preimage':'false','ttl':'1234'};")
.discard_result().then([&] {
assert_ext_correctness(e, cdc::cdc_extension{{{"enabled","true"},{"postimage","false"},{"preimage","false"},{"ttl","1234"}}});
});
});
}, cfg);
}