216 lines
7.8 KiB
C++
216 lines
7.8 KiB
C++
// Copyright 2024 Memgraph Ltd.
|
|
//
|
|
// Use of this software is governed by the Business Source License
|
|
// included in the file licenses/BSL.txt; by using this file, you agree to be bound by the terms of the Business Source
|
|
// License, and you may not use this file except in compliance with the Business Source License.
|
|
//
|
|
// As of the Change Date specified in that file, in accordance with
|
|
// the Business Source License, use of this software will be governed
|
|
// by the Apache License, Version 2.0, included in the file
|
|
// licenses/APL.txt.
|
|
|
|
#ifdef MG_ENTERPRISE
|
|
|
|
#include "nuraft/coordinator_state_machine.hpp"
|
|
#include "utils/logging.hpp"
|
|
|
|
namespace {
|
|
constexpr int MAX_SNAPSHOTS = 3;
|
|
} // namespace
|
|
|
|
namespace memgraph::coordination {
|
|
|
|
auto CoordinatorStateMachine::FindCurrentMainInstanceName() const -> std::optional<std::string> {
|
|
return cluster_state_.FindCurrentMainInstanceName();
|
|
}
|
|
|
|
auto CoordinatorStateMachine::MainExists() const -> bool { return cluster_state_.MainExists(); }
|
|
|
|
auto CoordinatorStateMachine::IsMain(std::string_view instance_name) const -> bool {
|
|
return cluster_state_.IsMain(instance_name);
|
|
}
|
|
|
|
auto CoordinatorStateMachine::IsReplica(std::string_view instance_name) const -> bool {
|
|
return cluster_state_.IsReplica(instance_name);
|
|
}
|
|
|
|
auto CoordinatorStateMachine::CreateLog(nlohmann::json &&log) -> ptr<buffer> {
|
|
auto const log_dump = log.dump();
|
|
ptr<buffer> log_buf = buffer::alloc(sizeof(uint32_t) + log_dump.size());
|
|
buffer_serializer bs(log_buf);
|
|
bs.put_str(log_dump);
|
|
return log_buf;
|
|
}
|
|
|
|
auto CoordinatorStateMachine::SerializeRegisterInstance(CoordinatorClientConfig const &config) -> ptr<buffer> {
|
|
return CreateLog({{"action", RaftLogAction::REGISTER_REPLICATION_INSTANCE}, {"info", config}});
|
|
}
|
|
|
|
auto CoordinatorStateMachine::SerializeUnregisterInstance(std::string_view instance_name) -> ptr<buffer> {
|
|
return CreateLog({{"action", RaftLogAction::UNREGISTER_REPLICATION_INSTANCE}, {"info", instance_name}});
|
|
}
|
|
|
|
auto CoordinatorStateMachine::SerializeSetInstanceAsMain(std::string_view instance_name) -> ptr<buffer> {
|
|
return CreateLog({{"action", RaftLogAction::SET_INSTANCE_AS_MAIN}, {"info", instance_name}});
|
|
}
|
|
|
|
auto CoordinatorStateMachine::SerializeSetInstanceAsReplica(std::string_view instance_name) -> ptr<buffer> {
|
|
return CreateLog({{"action", RaftLogAction::SET_INSTANCE_AS_REPLICA}, {"info", instance_name}});
|
|
}
|
|
|
|
auto CoordinatorStateMachine::SerializeUpdateUUID(utils::UUID const &uuid) -> ptr<buffer> {
|
|
return CreateLog({{"action", RaftLogAction::UPDATE_UUID}, {"info", uuid}});
|
|
}
|
|
|
|
auto CoordinatorStateMachine::DecodeLog(buffer &data) -> std::pair<TRaftLog, RaftLogAction> {
|
|
buffer_serializer bs(data);
|
|
auto const json = nlohmann::json::parse(bs.get_str());
|
|
|
|
auto const action = json["action"].get<RaftLogAction>();
|
|
auto const &info = json["info"];
|
|
|
|
switch (action) {
|
|
case RaftLogAction::REGISTER_REPLICATION_INSTANCE:
|
|
return {info.get<CoordinatorClientConfig>(), action};
|
|
case RaftLogAction::UPDATE_UUID:
|
|
return {info.get<utils::UUID>(), action};
|
|
case RaftLogAction::UNREGISTER_REPLICATION_INSTANCE:
|
|
case RaftLogAction::SET_INSTANCE_AS_MAIN:
|
|
[[fallthrough]];
|
|
case RaftLogAction::SET_INSTANCE_AS_REPLICA:
|
|
return {info.get<std::string>(), action};
|
|
}
|
|
throw std::runtime_error("Unknown action");
|
|
}
|
|
|
|
auto CoordinatorStateMachine::pre_commit(ulong const /*log_idx*/, buffer & /*data*/) -> ptr<buffer> { return nullptr; }
|
|
|
|
auto CoordinatorStateMachine::commit(ulong const log_idx, buffer &data) -> ptr<buffer> {
|
|
spdlog::debug("Commit: log_idx={}, data.size()={}", log_idx, data.size());
|
|
auto const [parsed_data, log_action] = DecodeLog(data);
|
|
cluster_state_.DoAction(parsed_data, log_action);
|
|
last_committed_idx_ = log_idx;
|
|
|
|
// Return raft log number
|
|
ptr<buffer> ret = buffer::alloc(sizeof(log_idx));
|
|
buffer_serializer bs_ret(ret);
|
|
bs_ret.put_u64(log_idx);
|
|
return ret;
|
|
}
|
|
|
|
auto CoordinatorStateMachine::commit_config(ulong const log_idx, ptr<cluster_config> & /*new_conf*/) -> void {
|
|
last_committed_idx_ = log_idx;
|
|
spdlog::debug("Commit config: log_idx={}", log_idx);
|
|
}
|
|
|
|
auto CoordinatorStateMachine::rollback(ulong const log_idx, buffer &data) -> void {
|
|
// NOTE: Nothing since we don't do anything in pre_commit
|
|
spdlog::debug("Rollback: log_idx={}, data.size()={}", log_idx, data.size());
|
|
}
|
|
|
|
auto CoordinatorStateMachine::read_logical_snp_obj(snapshot &snapshot, void *& /*user_snp_ctx*/, ulong obj_id,
|
|
ptr<buffer> &data_out, bool &is_last_obj) -> int {
|
|
spdlog::debug("read logical snapshot object, obj_id: {}", obj_id);
|
|
|
|
ptr<SnapshotCtx> ctx = nullptr;
|
|
{
|
|
auto ll = std::lock_guard{snapshots_lock_};
|
|
auto entry = snapshots_.find(snapshot.get_last_log_idx());
|
|
if (entry == snapshots_.end()) {
|
|
data_out = nullptr;
|
|
is_last_obj = true;
|
|
return 0;
|
|
}
|
|
ctx = entry->second;
|
|
}
|
|
|
|
if (obj_id == 0) {
|
|
// Object ID == 0: first object, put dummy data.
|
|
data_out = buffer::alloc(sizeof(int32));
|
|
buffer_serializer bs(data_out);
|
|
bs.put_i32(0);
|
|
is_last_obj = false;
|
|
} else {
|
|
// Object ID > 0: second object, put actual value.
|
|
ctx->cluster_state_.Serialize(data_out);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
auto CoordinatorStateMachine::save_logical_snp_obj(snapshot &snapshot, ulong &obj_id, buffer &data, bool is_first_obj,
|
|
bool is_last_obj) -> void {
|
|
spdlog::debug("save logical snapshot object, obj_id: {}, is_first_obj: {}, is_last_obj: {}", obj_id, is_first_obj,
|
|
is_last_obj);
|
|
|
|
if (obj_id == 0) {
|
|
ptr<buffer> snp_buf = snapshot.serialize();
|
|
auto ss = snapshot::deserialize(*snp_buf);
|
|
create_snapshot_internal(ss);
|
|
} else {
|
|
auto cluster_state = CoordinatorClusterState::Deserialize(data);
|
|
|
|
auto ll = std::lock_guard{snapshots_lock_};
|
|
auto entry = snapshots_.find(snapshot.get_last_log_idx());
|
|
DMG_ASSERT(entry != snapshots_.end());
|
|
entry->second->cluster_state_ = cluster_state;
|
|
}
|
|
}
|
|
|
|
auto CoordinatorStateMachine::apply_snapshot(snapshot &s) -> bool {
|
|
auto ll = std::lock_guard{snapshots_lock_};
|
|
spdlog::debug("apply snapshot, last_log_idx: {}", s.get_last_log_idx());
|
|
|
|
auto entry = snapshots_.find(s.get_last_log_idx());
|
|
if (entry == snapshots_.end()) return false;
|
|
|
|
cluster_state_ = entry->second->cluster_state_;
|
|
return true;
|
|
}
|
|
|
|
auto CoordinatorStateMachine::free_user_snp_ctx(void *&user_snp_ctx) -> void {}
|
|
|
|
auto CoordinatorStateMachine::last_snapshot() -> ptr<snapshot> {
|
|
auto ll = std::lock_guard{snapshots_lock_};
|
|
spdlog::debug("last_snapshot");
|
|
auto entry = snapshots_.rbegin();
|
|
if (entry == snapshots_.rend()) return nullptr;
|
|
|
|
ptr<SnapshotCtx> ctx = entry->second;
|
|
return ctx->snapshot_;
|
|
}
|
|
|
|
auto CoordinatorStateMachine::last_commit_index() -> ulong { return last_committed_idx_; }
|
|
|
|
auto CoordinatorStateMachine::create_snapshot(snapshot &s, async_result<bool>::handler_type &when_done) -> void {
|
|
spdlog::debug("create_snapshot, last_log_idx: {}", s.get_last_log_idx());
|
|
ptr<buffer> snp_buf = s.serialize();
|
|
ptr<snapshot> ss = snapshot::deserialize(*snp_buf);
|
|
create_snapshot_internal(ss);
|
|
|
|
ptr<std::exception> except(nullptr);
|
|
bool ret = true;
|
|
when_done(ret, except);
|
|
}
|
|
|
|
auto CoordinatorStateMachine::create_snapshot_internal(ptr<snapshot> snapshot) -> void {
|
|
auto ll = std::lock_guard{snapshots_lock_};
|
|
spdlog::debug("create_snapshot_internal, last_log_idx: {}", snapshot->get_last_log_idx());
|
|
|
|
auto ctx = cs_new<SnapshotCtx>(snapshot, cluster_state_);
|
|
snapshots_[snapshot->get_last_log_idx()] = ctx;
|
|
|
|
while (snapshots_.size() > MAX_SNAPSHOTS) {
|
|
snapshots_.erase(snapshots_.begin());
|
|
}
|
|
}
|
|
|
|
auto CoordinatorStateMachine::GetInstances() const -> std::vector<InstanceState> {
|
|
return cluster_state_.GetInstances();
|
|
}
|
|
|
|
auto CoordinatorStateMachine::GetUUID() const -> utils::UUID { return cluster_state_.GetUUID(); }
|
|
|
|
} // namespace memgraph::coordination
|
|
#endif
|