/* -*- c-basic-offset: 2 -*- */ /* Copyright(C) 2015-2017 Kouhei Sutou This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version. This library 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1335 USA */ #include #include #include #include "mrn_database_repairer.hpp" #include "mrn_path_mapper.hpp" // for debug #define MRN_CLASS_NAME "mrn::DatabaseRepairer" #include #include #include #ifndef WIN32 # include #endif namespace mrn { struct CheckResult { CheckResult() : is_crashed(false), is_corrupt(false) { } bool is_crashed; bool is_corrupt; }; DatabaseRepairer::DatabaseRepairer(grn_ctx *ctx, THD *thd) : ctx_(ctx), thd_(thd), base_directory_(NULL), base_directory_buffer_(), path_prefix_(NULL), path_prefix_buffer_(), path_prefix_length_(0), mrn_db_file_suffix_length_(strlen(MRN_DB_FILE_SUFFIX)) { } DatabaseRepairer::~DatabaseRepairer() = default; bool DatabaseRepairer::is_crashed(void) { MRN_DBUG_ENTER_METHOD(); CheckResult result; each_database(&DatabaseRepairer::check_body, &result); DBUG_RETURN(result.is_crashed); } bool DatabaseRepairer::is_corrupt(void) { MRN_DBUG_ENTER_METHOD(); CheckResult result; each_database(&DatabaseRepairer::check_body, &result); DBUG_RETURN(result.is_corrupt); } bool DatabaseRepairer::repair(void) { MRN_DBUG_ENTER_METHOD(); bool succeeded = true; each_database(&DatabaseRepairer::repair_body, &succeeded); DBUG_RETURN(succeeded); } void DatabaseRepairer::each_database(EachBodyFunc each_body_func, void *user_data) { MRN_DBUG_ENTER_METHOD(); detect_paths(); #ifdef WIN32 WIN32_FIND_DATA data; HANDLE finder = FindFirstFile(base_directory_, &data); if (finder == INVALID_HANDLE_VALUE) { DBUG_VOID_RETURN; } grn_ctx ctx; grn_rc rc = grn_ctx_init(&ctx, 0); if (rc == GRN_SUCCESS) { do { each_database_body(data.cFileName, &ctx, each_body_func, user_data); } while (FindNextFile(finder, &data) != 0); grn_ctx_fin(&ctx); } else { GRN_LOG(ctx_, GRN_LOG_WARNING, "[mroonga][database][repairer][each] " "failed to initialize grn_ctx: %d: %s", rc, grn_rc_to_string(rc)); } FindClose(finder); #else DIR *dir = opendir(base_directory_); if (!dir) { DBUG_VOID_RETURN; } grn_ctx ctx; grn_rc rc = grn_ctx_init(&ctx, 0); if (rc == GRN_SUCCESS) { while (struct dirent *entry = readdir(dir)) { each_database_body(entry->d_name, &ctx, each_body_func, user_data); } grn_ctx_fin(&ctx); } else { GRN_LOG(ctx_, GRN_LOG_WARNING, "[mroonga][database][repairer][each] " "failed to initialize grn_ctx: %d: %s", rc, grn_rc_to_string(rc)); } closedir(dir); #endif DBUG_VOID_RETURN; } void DatabaseRepairer::each_database_body(const char *base_path, grn_ctx *ctx, EachBodyFunc each_body_func, void *user_data) { MRN_DBUG_ENTER_METHOD(); if (path_prefix_length_ > 0 && strncmp(base_path, path_prefix_, path_prefix_length_) != 0) { DBUG_VOID_RETURN; } size_t path_length = strlen(base_path); if (path_length <= mrn_db_file_suffix_length_) { DBUG_VOID_RETURN; } if (strncmp(base_path + (path_length - mrn_db_file_suffix_length_), MRN_DB_FILE_SUFFIX, mrn_db_file_suffix_length_) != 0) { DBUG_VOID_RETURN; } char db_path[MRN_MAX_PATH_SIZE]; snprintf(db_path, MRN_MAX_PATH_SIZE, "%s%c%s", base_directory_, FN_LIBCHAR, base_path); grn_obj *db = grn_db_open(ctx, db_path); if (!db) { DBUG_VOID_RETURN; } (this->*each_body_func)(ctx, db, db_path, user_data); grn_obj_close(ctx, db); DBUG_VOID_RETURN; } void DatabaseRepairer::detect_paths(void) { MRN_DBUG_ENTER_METHOD(); const char *raw_path_prefix = mrn::PathMapper::default_path_prefix; if (!raw_path_prefix) { base_directory_ = "."; path_prefix_ = NULL; DBUG_VOID_RETURN; } strcpy(base_directory_buffer_, raw_path_prefix); size_t raw_path_prefix_length = strlen(raw_path_prefix); size_t separator_position = raw_path_prefix_length; for (; separator_position > 0; separator_position--) { if (mrn_is_directory_separator(base_directory_buffer_[separator_position])) { break; } } if (separator_position == 0 || separator_position == raw_path_prefix_length) { base_directory_ = "."; } else { base_directory_buffer_[separator_position] = '\0'; base_directory_ = base_directory_buffer_; strcpy(path_prefix_buffer_, raw_path_prefix + separator_position + 1); path_prefix_ = path_prefix_buffer_; path_prefix_length_ = strlen(path_prefix_); } DBUG_VOID_RETURN; } void DatabaseRepairer::check_body(grn_ctx *ctx, grn_obj *db, const char *db_path, void *user_data) { MRN_DBUG_ENTER_METHOD(); CheckResult *result = static_cast(user_data); if (grn_obj_is_locked(ctx, db)) { result->is_crashed = true; result->is_corrupt = true; DBUG_VOID_RETURN; } grn_table_cursor *cursor; cursor = grn_table_cursor_open(ctx, db, NULL, 0, NULL, 0, 0, -1, GRN_CURSOR_BY_ID); if (!cursor) { result->is_crashed = true; result->is_corrupt = true; DBUG_VOID_RETURN; } grn_id id; while ((id = grn_table_cursor_next(ctx, cursor)) != GRN_ID_NIL) { if (grn_id_is_builtin(ctx, id)) { continue; } grn_obj *object = grn_ctx_at(ctx, id); if (!object) { if (ctx->rc == GRN_SUCCESS) { continue; } else { result->is_corrupt = true; break; } } switch (object->header.type) { case GRN_TABLE_HASH_KEY : case GRN_TABLE_PAT_KEY: case GRN_TABLE_DAT_KEY: case GRN_TABLE_NO_KEY: case GRN_COLUMN_FIX_SIZE: case GRN_COLUMN_VAR_SIZE: case GRN_COLUMN_INDEX: if (grn_obj_is_locked(ctx_, object)) { result->is_crashed = true; result->is_corrupt = true; } break; default: break; } grn_obj_unlink(ctx, object); if (result->is_crashed || result->is_corrupt) { break; } } grn_table_cursor_close(ctx, cursor); DBUG_VOID_RETURN; } void DatabaseRepairer::repair_body(grn_ctx *ctx, grn_obj *db, const char *db_path, void *user_data) { MRN_DBUG_ENTER_METHOD(); bool *succeeded = static_cast(user_data); if (grn_db_recover(ctx, db) != GRN_SUCCESS) { push_warning_printf(thd_, MRN_SEVERITY_WARNING, ER_NOT_KEYFILE, "mroonga: repair: " "Failed to recover database: <%s>: <%s>", db_path, ctx->errbuf); *succeeded = false; } DBUG_VOID_RETURN; } }