diff --git a/meilisearch-core/src/database.rs b/meilisearch-core/src/database.rs index b9e952616..5510331a1 100644 --- a/meilisearch-core/src/database.rs +++ b/meilisearch-core/src/database.rs @@ -3,6 +3,7 @@ use std::fs::File; use std::path::Path; use std::sync::{Arc, RwLock}; use std::{fs, thread}; +use std::io::{Read, Write, ErrorKind}; use chrono::{DateTime, Utc}; use crossbeam_channel::{Receiver, Sender}; @@ -161,11 +162,62 @@ fn update_awaiter( Ok(()) } +/// Ensures Meilisearch version is compatible with the database, returns an error versions mismatch. +/// If create is set to true, a VERSION file is created with the current version. +fn version_guard(path: &Path, create: bool) -> MResult<()> { + let current_version_major = env!("CARGO_PKG_VERSION_MAJOR"); + let current_version_minor = env!("CARGO_PKG_VERSION_MINOR"); + let current_version_patch = env!("CARGO_PKG_VERSION_PATCH"); + let version_path = path.join("VERSION"); + + match File::open(&version_path) { + Ok(mut file) => { + let mut version = String::new(); + file.read_to_string(&mut version)?; + let mut version = version.split("."); + + let version_major = version.next().ok_or(Error::VersionMismatch("bad VERSION file".to_string()))?; + let version_minor = version.next().ok_or(Error::VersionMismatch("bad VERSION file".to_string()))?; + + if version_major != current_version_major || version_minor != current_version_minor { + return Err(Error::VersionMismatch(format!("{}.{}.XX", version_major, version_major))); + } + } + Err(error) => { + match error.kind() { + ErrorKind::NotFound => { + if create { + // when no version file is found, and we've beem told to create one, + // create a new file wioth the current version in it. + let mut version_file = File::create(&version_path)?; + version_file.write_all(format!("{}.{}.{}", + current_version_major, + current_version_minor, + current_version_patch).as_bytes())?; + } else { + // when no version file is found and we were not told to create one, this + // means that the version is inferior to the one this feature was added. + return Err(Error::VersionMismatch(format!("<0.12.0"))); + } + } + _ => return Err(error.into()) + } + } + } + Ok(()) +} + impl Database { pub fn open_or_create(path: impl AsRef, options: DatabaseOptions) -> MResult { let main_path = path.as_ref().join("main"); let update_path = path.as_ref().join("update"); + //create db directory + fs::create_dir_all(&path)?; + + // create file only if main db wasn't created before (first run) + version_guard(path.as_ref(), !main_path.exists() && !update_path.exists())?; + fs::create_dir_all(&main_path)?; let env = heed::EnvOpenOptions::new() .map_size(options.main_map_size) diff --git a/meilisearch-core/src/error.rs b/meilisearch-core/src/error.rs index 477063bac..379dbd8ff 100644 --- a/meilisearch-core/src/error.rs +++ b/meilisearch-core/src/error.rs @@ -15,22 +15,23 @@ pub type MResult = Result; #[derive(Debug)] pub enum Error { - Io(io::Error), - IndexAlreadyExists, - MissingPrimaryKey, - SchemaMissing, - WordIndexMissing, - MissingDocumentId, - MaxFieldsLimitExceeded, - Schema(meilisearch_schema::Error), - Heed(heed::Error), - Fst(fst::Error), - SerdeJson(SerdeJsonError), Bincode(bincode::Error), - Serializer(SerializerError), Deserializer(DeserializerError), - FilterParseError(PestError), FacetError(FacetError), + FilterParseError(PestError), + Fst(fst::Error), + Heed(heed::Error), + IndexAlreadyExists, + Io(io::Error), + MaxFieldsLimitExceeded, + MissingDocumentId, + MissingPrimaryKey, + Schema(meilisearch_schema::Error), + SchemaMissing, + SerdeJson(SerdeJsonError), + Serializer(SerializerError), + VersionMismatch(String), + WordIndexMissing, } impl ErrorCode for Error { @@ -53,6 +54,7 @@ impl ErrorCode for Error { | Bincode(_) | Serializer(_) | Deserializer(_) + | VersionMismatch(_) | Io(_) => Code::Internal, } } @@ -141,22 +143,23 @@ impl fmt::Display for Error { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { use self::Error::*; match self { - Io(e) => write!(f, "{}", e), - IndexAlreadyExists => write!(f, "index already exists"), - MissingPrimaryKey => write!(f, "schema cannot be built without a primary key"), - SchemaMissing => write!(f, "this index does not have a schema"), - WordIndexMissing => write!(f, "this index does not have a word index"), - MissingDocumentId => write!(f, "document id is missing"), - MaxFieldsLimitExceeded => write!(f, "maximum number of fields in a document exceeded"), - Schema(e) => write!(f, "schema error; {}", e), - Heed(e) => write!(f, "heed error; {}", e), - Fst(e) => write!(f, "fst error; {}", e), - SerdeJson(e) => write!(f, "serde json error; {}", e), Bincode(e) => write!(f, "bincode error; {}", e), - Serializer(e) => write!(f, "serializer error; {}", e), Deserializer(e) => write!(f, "deserializer error; {}", e), - FilterParseError(e) => write!(f, "error parsing filter; {}", e), FacetError(e) => write!(f, "error processing facet filter: {}", e), + FilterParseError(e) => write!(f, "error parsing filter; {}", e), + Fst(e) => write!(f, "fst error; {}", e), + Heed(e) => write!(f, "heed error; {}", e), + IndexAlreadyExists => write!(f, "index already exists"), + Io(e) => write!(f, "{}", e), + MaxFieldsLimitExceeded => write!(f, "maximum number of fields in a document exceeded"), + MissingDocumentId => write!(f, "document id is missing"), + MissingPrimaryKey => write!(f, "schema cannot be built without a primary key"), + Schema(e) => write!(f, "schema error; {}", e), + SchemaMissing => write!(f, "this index does not have a schema"), + SerdeJson(e) => write!(f, "serde json error; {}", e), + Serializer(e) => write!(f, "serializer error; {}", e), + VersionMismatch(version) => write!(f, "Cannot open database, expected Meilisearch version: {}", version), + WordIndexMissing => write!(f, "this index does not have a word index"), } } } diff --git a/meilisearch-http/src/data.rs b/meilisearch-http/src/data.rs index 3692b0b69..04151d3cb 100644 --- a/meilisearch-http/src/data.rs +++ b/meilisearch-http/src/data.rs @@ -4,6 +4,7 @@ use std::sync::Arc; use meilisearch_core::{Database, DatabaseOptions}; use sha2::Digest; use sysinfo::Pid; +use log::error; use crate::index_update_callback; use crate::option::Opt; @@ -66,7 +67,13 @@ impl Data { let http_payload_size_limit = opt.http_payload_size_limit; - let db = Arc::new(Database::open_or_create(opt.db_path, db_opt).unwrap()); + let db = match Database::open_or_create(opt.db_path, db_opt) { + Ok(db) => Arc::new(db), + Err(e) => { + error!("{}", e); + std::process::exit(1); + } + }; let mut api_keys = ApiKeys { master: opt.master_key,