Add durability for text indices

This commit is contained in:
Ante Pušić 2024-02-05 09:37:45 +01:00
parent ea64f2e906
commit 4b46f1bb54
5 changed files with 148 additions and 3 deletions

View File

@ -1,4 +1,4 @@
// Copyright 2023 Memgraph Ltd. // Copyright 2024 Memgraph Ltd.
// //
// Use of this software is governed by the Business Source License // 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 // included in the file licenses/BSL.txt; by using this file, you agree to be bound by the terms of the Business Source
@ -197,6 +197,21 @@ void RecoverIndicesAndStats(const RecoveredIndicesAndConstraints::IndicesMetadat
} }
spdlog::info("Label+property indices statistics are recreated."); spdlog::info("Label+property indices statistics are recreated.");
if (flags::run_time::GetTextSearchEnabled()) {
// Recover text indices.
spdlog::info("Recreating {} text indices from metadata.", indices_metadata.text.size());
auto *mem_text_index = static_cast<TextIndex *>(indices->text_index_.get());
for (const auto &item : indices_metadata.text) {
const auto index_name = item.first;
const auto label_id = item.second;
if (!mem_text_index->RecoverIndex(item.first, item.second, vertices->access(), name_id_mapper))
throw RecoveryFailure("The text index must be created here!");
spdlog::info("Text index {} on :{} is recreated from metadata", item.first,
name_id_mapper->IdToName(item.second.AsUint()));
}
}
spdlog::info("Indices are recreated."); spdlog::info("Indices are recreated.");
spdlog::info("Recreating constraints from metadata."); spdlog::info("Recreating constraints from metadata.");

View File

@ -13,6 +13,7 @@
#include <thread> #include <thread>
#include "flags/run_time_configurable.hpp"
#include "spdlog/spdlog.h" #include "spdlog/spdlog.h"
#include "storage/v2/durability/exceptions.hpp" #include "storage/v2/durability/exceptions.hpp"
#include "storage/v2/durability/paths.hpp" #include "storage/v2/durability/paths.hpp"
@ -799,7 +800,6 @@ RecoveredSnapshot LoadSnapshotVersion14(const std::filesystem::path &path, utils
SPDLOG_TRACE("Recovered property \"{}\" with value \"{}\" for vertex {}.", SPDLOG_TRACE("Recovered property \"{}\" with value \"{}\" for vertex {}.",
name_id_mapper->IdToName(snapshot_id_map.at(*key)), *value, *gid); name_id_mapper->IdToName(snapshot_id_map.at(*key)), *value, *gid);
props.SetProperty(get_property_from_id(*key), *value); props.SetProperty(get_property_from_id(*key), *value);
// TODO antepusic: update text index here or at the end of LoadSnapshot()?
} }
} }
@ -993,6 +993,24 @@ RecoveredSnapshot LoadSnapshotVersion14(const std::filesystem::path &path, utils
spdlog::info("Metadata of label+property indices are recovered."); spdlog::info("Metadata of label+property indices are recovered.");
} }
// Recover text indices.
if (flags::run_time::GetTextSearchEnabled()) {
auto size = snapshot.ReadUint();
if (!size) throw RecoveryFailure("Couldn't recover the number of text indices!");
spdlog::info("Recovering metadata of {} text indices.", *size);
for (uint64_t i = 0; i < *size; ++i) {
auto index_name = snapshot.ReadString();
if (!index_name.has_value()) throw RecoveryFailure("Couldn't read text index name!");
auto label = snapshot.ReadUint();
if (!label) throw RecoveryFailure("Couldn't read text index label!");
AddRecoveredIndexConstraint(&indices_constraints.indices.text, {index_name.value(), get_label_from_id(*label)},
"The text index already exists!");
SPDLOG_TRACE("Recovered metadata of text index {} for :{}", index_name.value(),
name_id_mapper->IdToName(snapshot_id_map.at(*label)));
}
spdlog::info("Metadata of label+property indices are recovered.");
}
spdlog::info("Metadata of indices are recovered."); spdlog::info("Metadata of indices are recovered.");
} }
@ -2106,6 +2124,16 @@ void CreateSnapshot(Storage *storage, Transaction *transaction, const std::files
snapshot.SetPosition(last_pos); snapshot.SetPosition(last_pos);
} }
} }
// Write text indices.
if (flags::run_time::GetTextSearchEnabled()) {
auto text = storage->indices_.text_index_->ListIndices();
snapshot.WriteUint(text.size());
for (const auto &item : text) {
snapshot.WriteString(item.first);
write_mapping(item.second);
}
}
} }
// Write constraints. // Write constraints.

View File

@ -802,7 +802,6 @@ RecoveryInfo LoadWal(const std::filesystem::path &path, RecoveredIndicesAndConst
auto &property_value = delta.vertex_edge_set_property.value; auto &property_value = delta.vertex_edge_set_property.value;
vertex->properties.SetProperty(property_id, property_value); vertex->properties.SetProperty(property_id, property_value);
// TODO antepusic: update text index here or at the end of LoadWal()?
break; break;
} }

View File

@ -292,6 +292,105 @@ bool TextIndex::CreateIndex(std::string index_name, LabelId label, memgraph::que
return true; return true;
} }
bool TextIndex::RecoverIndex(std::string index_name, LabelId label,
memgraph::utils::SkipList<Vertex>::Accessor vertices, NameIdMapper *name_id_mapper) {
if (!flags::run_time::GetTextSearchEnabled()) {
throw query::QueryException("To use text indices, enable the text search feature.");
}
nlohmann::json mappings = {};
mappings["properties"] = {};
mappings["properties"]["metadata"] = {{"type", "json"}, {"fast", true}, {"stored", true}, {"text", true}};
mappings["properties"]["data"] = {{"type", "json"}, {"fast", true}, {"stored", true}, {"text", true}};
try {
index_.emplace(index_name,
TextIndexData{.context_ = mgcxx::text_search::create_index(
index_name, mgcxx::text_search::IndexConfig{.mappings = mappings.dump()}),
.scope_ = label});
} catch (const std::exception &e) {
throw query::QueryException(fmt::format("Tantivy error: {}", e.what()));
}
label_to_index_.emplace(label, index_name);
bool has_schema = false;
std::vector<std::pair<PropertyId, std::string>> indexed_properties{};
auto &index_context = index_.at(index_name).context_;
for (const auto &v : vertices) {
if (std::find(v.labels.begin(), v.labels.end(), label) == v.labels.end()) {
continue;
}
auto vertex_properties = v.properties.Properties();
if (!has_schema) [[unlikely]] {
for (const auto &[prop_id, prop_val] : vertex_properties) {
if (prop_val.IsBool() || prop_val.IsInt() || prop_val.IsDouble() || prop_val.IsString()) {
indexed_properties.emplace_back(
std::pair<PropertyId, std::string>{prop_id, name_id_mapper->IdToName(prop_id.AsUint())});
}
}
has_schema = true;
}
nlohmann::json document = {};
nlohmann::json properties = {};
for (const auto &[prop_id, prop_name] : indexed_properties) {
if (!vertex_properties.contains(prop_id)) {
continue;
}
const auto prop_value = vertex_properties.at(prop_id);
switch (prop_value.type()) {
case PropertyValue::Type::Bool:
properties[prop_name] = prop_value.ValueBool();
break;
case PropertyValue::Type::Int:
properties[prop_name] = prop_value.ValueInt();
break;
case PropertyValue::Type::Double:
properties[prop_name] = prop_value.ValueDouble();
break;
case PropertyValue::Type::String:
properties[prop_name] = prop_value.ValueString();
break;
case PropertyValue::Type::Null:
case PropertyValue::Type::List:
case PropertyValue::Type::Map:
case PropertyValue::Type::TemporalData:
default:
continue;
}
}
document["data"] = properties;
document["metadata"] = {};
document["metadata"]["gid"] = v.gid.AsInt();
document["metadata"]["txid"] = -1;
document["metadata"]["deleted"] = false;
document["metadata"]["is_node"] = true;
try {
mgcxx::text_search::add_document(
index_context,
mgcxx::text_search::DocumentInput{
.data = document.dump(-1, ' ', false, nlohmann::json::error_handler_t::replace)},
kDoSkipCommit);
} catch (const std::exception &e) {
throw query::QueryException(fmt::format("Tantivy error: {}", e.what()));
}
}
// As CREATE TEXT INDEX (...) queries dont accumulate deltas, db_transactional_accessor_->Commit() does not reach
// the code area where changes to indices are committed. To get around that without needing to commit text indices
// after every such query, we commit here.
try {
mgcxx::text_search::commit(index_context);
} catch (const std::exception &e) {
throw query::QueryException(fmt::format("Tantivy error: {}", e.what()));
}
return true;
}
bool TextIndex::DropIndex(std::string index_name) { bool TextIndex::DropIndex(std::string index_name) {
if (!flags::run_time::GetTextSearchEnabled()) { if (!flags::run_time::GetTextSearchEnabled()) {
throw query::QueryException("To use text indices, enable the text search feature."); throw query::QueryException("To use text indices, enable the text search feature.");

View File

@ -12,6 +12,7 @@
#pragma once #pragma once
#include "storage/v2/id_types.hpp" #include "storage/v2/id_types.hpp"
#include "storage/v2/name_id_mapper.hpp"
#include "storage/v2/transaction.hpp" #include "storage/v2/transaction.hpp"
#include "storage/v2/vertex.hpp" #include "storage/v2/vertex.hpp"
#include "text_search.hpp" #include "text_search.hpp"
@ -75,6 +76,9 @@ class TextIndex {
bool CreateIndex(std::string index_name, LabelId label, memgraph::query::DbAccessor *db); bool CreateIndex(std::string index_name, LabelId label, memgraph::query::DbAccessor *db);
bool RecoverIndex(std::string index_name, LabelId label, memgraph::utils::SkipList<Vertex>::Accessor vertices,
NameIdMapper *name_id_mapper);
bool DropIndex(std::string index_name); bool DropIndex(std::string index_name);
bool IndexExists(std::string index_name) const; bool IndexExists(std::string index_name) const;