1
0
mirror of https://github.com/codership/wsrep-lib.git synced 2025-07-28 20:02:00 +03:00

Fixes to local streaming replication processing.

This commit is contained in:
Teemu Ollakka
2018-07-08 15:27:49 +03:00
parent 2913aecebd
commit 7c424d8337
12 changed files with 122 additions and 75 deletions

View File

@ -12,7 +12,7 @@ namespace db
{ {
class storage_service : public wsrep::storage_service class storage_service : public wsrep::storage_service
{ {
int start_transaction() override int start_transaction(const wsrep::ws_handle&) override
{ throw wsrep::not_implemented_error(); } { throw wsrep::not_implemented_error(); }
int append_fragment(const wsrep::id&, int append_fragment(const wsrep::id&,
wsrep::transaction_id, wsrep::transaction_id,

View File

@ -307,6 +307,12 @@ namespace wsrep
fragment_unit, fragment_unit,
size_t fragment_size); size_t fragment_size);
void disable_streaming()
{
assert(state_ == s_exec && mode_ == m_local);
transaction_.streaming_context_.disable();
}
/** /**
* Prepare write set meta data for fragment storage ordering. * Prepare write set meta data for fragment storage ordering.
* This method should be called from storage service commit * This method should be called from storage service commit
@ -316,7 +322,6 @@ namespace wsrep
const wsrep::ws_meta& ws_meta, const wsrep::ws_meta& ws_meta,
bool is_commit) bool is_commit)
{ {
assert(mode_ == m_high_priority);
assert(state_ == s_exec); assert(state_ == s_exec);
return transaction_.prepare_for_fragment_ordering( return transaction_.prepare_for_fragment_ordering(
ws_handle, ws_meta, is_commit); ws_handle, ws_meta, is_commit);

View File

@ -35,12 +35,11 @@ namespace wsrep
/** /**
* Start a new transaction for storage access. * Start a new transaction for storage access.
* *
* @param ws_hande Write set handle * @param[out] ws_handle Write set handle for a new transaction
* @param ws_meta Write set meta data
* *
* @return Zero in case of success, non-zero on error. * @return Zero in case of success, non-zero on error.
*/ */
virtual int start_transaction() = 0; virtual int start_transaction(const wsrep::ws_handle&) = 0;
/** /**
* Append fragment into stable storage. * Append fragment into stable storage.

View File

@ -46,9 +46,10 @@ namespace wsrep
fragment_size_ = 0; fragment_size_ = 0;
} }
void certified(wsrep::seqno seqno) void certified(wsrep::seqno seqno, size_t bytes)
{ {
fragments_.push_back(seqno); fragments_.push_back(seqno);
bytes_certified_ += bytes;
} }
size_t fragments_certified() const size_t fragments_certified() const

View File

@ -463,6 +463,20 @@ void wsrep::server_state::on_view(const wsrep::view& view)
if (view.status() == wsrep::view::primary) if (view.status() == wsrep::view::primary)
{ {
wsrep::unique_lock<wsrep::mutex> lock(mutex_); wsrep::unique_lock<wsrep::mutex> lock(mutex_);
if (view.own_index() >= 0)
{
if (id_.is_undefined())
{
// No identifier was passed during server state initialization
// and the ID was generated by the provider.
id_ = view.members()[view.own_index()].id();
}
else
{
// Own identifier must not change between views.
// assert(id_ == view.members()[view.own_index()].id());
}
}
assert(view.final() == false); assert(view.final() == false);
// //
@ -555,6 +569,7 @@ void wsrep::server_state::on_view(const wsrep::view& view)
assert(view.final()); assert(view.final());
wsrep::unique_lock<wsrep::mutex> lock(mutex_); wsrep::unique_lock<wsrep::mutex> lock(mutex_);
state(lock, s_disconnected); state(lock, s_disconnected);
id_ = wsrep::id::undefined();
} }
} }

View File

@ -64,9 +64,8 @@ namespace
~scoped_storage_service() ~scoped_storage_service()
{ {
storage_service_->reset_globals();
client_service_.store_globals();
deleter_(storage_service_); deleter_(storage_service_);
client_service_.store_globals();
} }
private: private:
scoped_storage_service(const scoped_storage_service&); scoped_storage_service(const scoped_storage_service&);
@ -666,8 +665,15 @@ int wsrep::transaction::after_statement()
void wsrep::transaction::after_applying() void wsrep::transaction::after_applying()
{ {
wsrep::unique_lock<wsrep::mutex> lock(client_state_.mutex_); wsrep::unique_lock<wsrep::mutex> lock(client_state_.mutex_);
assert(state_ == s_committed || state_ == s_aborted); debug_log_state("after_applying enter");
cleanup(); assert(state_ == s_executing ||
state_ == s_committed ||
state_ == s_aborted);
if (state_ != s_executing)
{
cleanup();
}
debug_log_state("after_applying leave");
} }
bool wsrep::transaction::bf_abort( bool wsrep::transaction::bf_abort(
@ -820,60 +826,80 @@ int wsrep::transaction::certify_fragment(
return 1; return 1;
} }
// The rest of this method will be executed in storage service if (provider().append_data(ws_handle_,
// scope. wsrep::const_buffer(data.data(), data.size())))
scoped_storage_service<storage_service_deleter>
sr_scope(
client_service_,
server_service_.storage_service(client_service_),
storage_service_deleter(server_service_));
wsrep::storage_service& storage_service(
sr_scope.storage_service());
// First the fragment is appended to the stable storage.
// This is done to ensure that there is enough capacity
// available to store the fragment. The fragment meta data
// is updated after certification.
if (storage_service.start_transaction())
{ {
lock.lock(); lock.lock();
state(lock, s_must_abort); state(lock, s_must_abort);
client_state_.override_error(wsrep::e_append_fragment_error);
return 1; return 1;
} }
if (storage_service.append_fragment(
client_state_.server_state().id(),
id(),
flags_,
wsrep::const_buffer(data.data(), data.size())))
{
lock.lock();
state(lock, s_must_abort);
client_state_.override_error(wsrep::e_append_fragment_error);
return 1;
}
enum wsrep::provider::status
cert_ret(provider().certify(client_state_.id().get(),
ws_handle_,
flags_,
ws_meta_));
int ret(0); int ret(0);
switch (cert_ret) // Storage service scope
{ {
case wsrep::provider::success: scoped_storage_service<storage_service_deleter>
streaming_context_.certified(ws_meta_.seqno()); sr_scope(
if (storage_service.commit(ws_handle_, ws_meta_)) client_service_,
server_service_.storage_service(client_service_),
storage_service_deleter(server_service_));
wsrep::storage_service& storage_service(
sr_scope.storage_service());
// First the fragment is appended to the stable storage.
// This is done to ensure that there is enough capacity
// available to store the fragment. The fragment meta data
// is updated after certification.
if (storage_service.start_transaction(ws_handle_))
{ {
ret = 1; lock.lock();
state(lock, s_must_abort);
client_state_.override_error(wsrep::e_append_fragment_error);
return 1;
} }
break;
default: wsrep::id server_id(client_state_.server_state().id());
storage_service.rollback(ws_handle_, ws_meta_); assert(server_id.is_undefined() == false);
ret = 1; if (storage_service.append_fragment(
break; server_id,
id(),
flags_,
wsrep::const_buffer(data.data(), data.size())))
{
lock.lock();
state(lock, s_must_abort);
client_state_.override_error(wsrep::e_append_fragment_error);
return 1;
}
wsrep::ws_meta sr_ws_meta;
enum wsrep::provider::status
cert_ret(provider().certify(client_state_.id().get(),
ws_handle_,
flags_,
sr_ws_meta));
switch (cert_ret)
{
case wsrep::provider::success:
assert(sr_ws_meta.seqno().is_undefined() == false);
streaming_context_.certified(sr_ws_meta.seqno(), data.size());
if (storage_service.update_fragment_meta(sr_ws_meta))
{
ret = 1;
break;
}
wsrep::log_info() << "Committing "
<< sr_ws_meta.transaction_id().get();
if (storage_service.commit(ws_handle_, sr_ws_meta))
{
ret = 1;
}
break;
default:
storage_service.rollback(ws_handle_, sr_ws_meta);
ret = 1;
break;
}
provider().release(ws_handle_);
} }
lock.lock(); lock.lock();
if (ret) if (ret)

View File

@ -64,5 +64,5 @@ int wsrep::mock_high_priority_service::apply_toi(const wsrep::ws_meta&,
void wsrep::mock_high_priority_service::after_apply() void wsrep::mock_high_priority_service::after_apply()
{ {
client_state_->after_statement(); client_state_->after_applying();
} }

View File

@ -28,29 +28,30 @@ wsrep::mock_storage_service::~mock_storage_service()
client_state_.cleanup(); client_state_.cleanup();
} }
int wsrep::mock_storage_service::start_transaction() int wsrep::mock_storage_service::start_transaction(const wsrep::ws_handle& ws_handle)
{ {
return client_state_.start_transaction( return client_state_.start_transaction(ws_handle.transaction_id());
server_state_.next_transaction_id());
} }
int wsrep::mock_storage_service::commit(const wsrep::ws_handle& ws_handle, int wsrep::mock_storage_service::commit(const wsrep::ws_handle& ws_handle,
const wsrep::ws_meta& ws_meta) const wsrep::ws_meta& ws_meta)
{ {
return client_state_.prepare_for_fragment_ordering( int ret(client_state_.prepare_for_fragment_ordering(
ws_handle, ws_meta, true) || ws_handle, ws_meta, true) ||
client_state_.before_commit() || client_state_.before_commit() ||
client_state_.ordered_commit() || client_state_.ordered_commit() ||
client_state_.after_commit() || client_state_.after_commit());
client_state_.after_statement(); client_state_.after_applying();
return ret;
} }
int wsrep::mock_storage_service::rollback(const wsrep::ws_handle& ws_handle, int wsrep::mock_storage_service::rollback(const wsrep::ws_handle& ws_handle,
const wsrep::ws_meta& ws_meta) const wsrep::ws_meta& ws_meta)
{ {
return client_state_.prepare_for_fragment_ordering( int ret(client_state_.prepare_for_fragment_ordering(
ws_handle, ws_meta, false) || ws_handle, ws_meta, false) ||
client_state_.before_rollback() || client_state_.before_rollback() ||
client_state_.after_rollback() || client_state_.after_rollback());
client_state_.after_statement(); client_state_.after_applying();
return ret;
} }

View File

@ -18,7 +18,7 @@ class mock_server_state;
wsrep::client_id); wsrep::client_id);
~mock_storage_service(); ~mock_storage_service();
int start_transaction() WSREP_OVERRIDE; int start_transaction(const wsrep::ws_handle&) WSREP_OVERRIDE;
int append_fragment(const wsrep::id&, int append_fragment(const wsrep::id&,
wsrep::transaction_id, wsrep::transaction_id,

View File

@ -172,7 +172,7 @@ BOOST_FIXTURE_TEST_CASE(server_state_sst_first_boostrap,
wsrep::id cluster_id("1"); wsrep::id cluster_id("1");
wsrep::gtid state_id(cluster_id, wsrep::seqno(0)); wsrep::gtid state_id(cluster_id, wsrep::seqno(0));
std::vector<wsrep::view::member> members; std::vector<wsrep::view::member> members;
members.push_back(wsrep::view::member(wsrep::id("1"), "name", "")); members.push_back(wsrep::view::member(wsrep::id("s1"), "name", ""));
wsrep::view bootstrap_view(state_id, wsrep::view bootstrap_view(state_id,
wsrep::seqno(1), wsrep::seqno(1),
wsrep::view::primary, wsrep::view::primary,
@ -198,7 +198,7 @@ BOOST_FIXTURE_TEST_CASE(server_state_init_first_boostrap,
wsrep::id cluster_id("1"); wsrep::id cluster_id("1");
wsrep::gtid state_id(cluster_id, wsrep::seqno(0)); wsrep::gtid state_id(cluster_id, wsrep::seqno(0));
std::vector<wsrep::view::member> members; std::vector<wsrep::view::member> members;
members.push_back(wsrep::view::member(wsrep::id("1"), "name", "")); members.push_back(wsrep::view::member(wsrep::id("s1"), "name", ""));
wsrep::view bootstrap_view(state_id, wsrep::view bootstrap_view(state_id,
wsrep::seqno(1), wsrep::seqno(1),
wsrep::view::primary, wsrep::view::primary,

View File

@ -950,7 +950,7 @@ BOOST_FIXTURE_TEST_CASE(transaction_1pc_applying,
BOOST_REQUIRE(tc.state() == wsrep::transaction::s_ordered_commit); BOOST_REQUIRE(tc.state() == wsrep::transaction::s_ordered_commit);
BOOST_REQUIRE(cc.after_commit() == 0); BOOST_REQUIRE(cc.after_commit() == 0);
BOOST_REQUIRE(tc.state() == wsrep::transaction::s_committed); BOOST_REQUIRE(tc.state() == wsrep::transaction::s_committed);
cc.after_statement(); cc.after_applying();
BOOST_REQUIRE(tc.state() == wsrep::transaction::s_committed); BOOST_REQUIRE(tc.state() == wsrep::transaction::s_committed);
BOOST_REQUIRE(tc.active() == false); BOOST_REQUIRE(tc.active() == false);
BOOST_REQUIRE(cc.current_error() == wsrep::e_success); BOOST_REQUIRE(cc.current_error() == wsrep::e_success);
@ -964,7 +964,7 @@ BOOST_FIXTURE_TEST_CASE(transaction_applying_rollback,
BOOST_REQUIRE(tc.state() == wsrep::transaction::s_aborting); BOOST_REQUIRE(tc.state() == wsrep::transaction::s_aborting);
BOOST_REQUIRE(cc.after_rollback() == 0); BOOST_REQUIRE(cc.after_rollback() == 0);
BOOST_REQUIRE(tc.state() == wsrep::transaction::s_aborted); BOOST_REQUIRE(tc.state() == wsrep::transaction::s_aborted);
cc.after_statement(); cc.after_applying();
BOOST_REQUIRE(tc.state() == wsrep::transaction::s_aborted); BOOST_REQUIRE(tc.state() == wsrep::transaction::s_aborted);
BOOST_REQUIRE(tc.active() == false); BOOST_REQUIRE(tc.active() == false);
BOOST_REQUIRE(cc.current_error() == wsrep::e_success); BOOST_REQUIRE(cc.current_error() == wsrep::e_success);

View File

@ -239,7 +239,7 @@ BOOST_FIXTURE_TEST_CASE(transaction_2pc_applying,
BOOST_REQUIRE(tc.state() == wsrep::transaction::s_ordered_commit); BOOST_REQUIRE(tc.state() == wsrep::transaction::s_ordered_commit);
BOOST_REQUIRE(cc.after_commit() == 0); BOOST_REQUIRE(cc.after_commit() == 0);
BOOST_REQUIRE(tc.state() == wsrep::transaction::s_committed); BOOST_REQUIRE(tc.state() == wsrep::transaction::s_committed);
BOOST_REQUIRE(cc.after_statement() == 0); cc.after_applying();
BOOST_REQUIRE(tc.state() == wsrep::transaction::s_committed); BOOST_REQUIRE(tc.state() == wsrep::transaction::s_committed);
BOOST_REQUIRE(tc.active() == false); BOOST_REQUIRE(tc.active() == false);
BOOST_REQUIRE(cc.current_error() == wsrep::e_success); BOOST_REQUIRE(cc.current_error() == wsrep::e_success);