From 4b7b2d6a906a5fc1f737bf014a4925b387873fef Mon Sep 17 00:00:00 2001 From: Tamo Date: Tue, 31 Jan 2023 12:24:37 +0100 Subject: [PATCH] fix the import of dump v2 generated by meilisearch v0.22.0 --- ...ompat__v1_to_v2__test__compat_v1_v2-3.snap | 1 + ...ompat__v1_to_v2__test__compat_v1_v2-6.snap | 6 +- ...ompat__v1_to_v2__test__compat_v1_v2-9.snap | 1 + dump/src/reader/compat/v1_to_v2.rs | 96 ++++--- dump/src/reader/compat/v2_to_v3.rs | 30 ++- dump/src/reader/mod.rs | 78 +++++- ...dump__reader__test__import_dump_v1-11.snap | 1 + .../dump__reader__test__import_dump_v1-5.snap | 1 + .../dump__reader__test__import_dump_v1-8.snap | 4 + ...rom_meilisearch_v0_22_0_issue_3435-11.snap | 25 ++ ...from_meilisearch_v0_22_0_issue_3435-5.snap | 39 +++ ...from_meilisearch_v0_22_0_issue_3435-8.snap | 30 +++ dump/src/reader/v2/mod.rs | 78 ++++++ dump/src/reader/v2/settings.rs | 248 ++++++++++++------ ...rom_meilisearch_v0_22_0_issue_3435-10.snap | 25 ++ ...from_meilisearch_v0_22_0_issue_3435-4.snap | 39 +++ ...from_meilisearch_v0_22_0_issue_3435-7.snap | 30 +++ dump/tests/assets/v2-v0.22.0.dump | Bin 0 -> 9809 bytes meilisearch/tests/dumps/mod.rs | 6 +- 19 files changed, 584 insertions(+), 154 deletions(-) create mode 100644 dump/src/reader/snapshots/dump__reader__test__import_dump_v2_from_meilisearch_v0_22_0_issue_3435-11.snap create mode 100644 dump/src/reader/snapshots/dump__reader__test__import_dump_v2_from_meilisearch_v0_22_0_issue_3435-5.snap create mode 100644 dump/src/reader/snapshots/dump__reader__test__import_dump_v2_from_meilisearch_v0_22_0_issue_3435-8.snap create mode 100644 dump/src/reader/v2/snapshots/dump__reader__v2__test__read_dump_v2_from_meilisearch_v0_22_0_issue_3435-10.snap create mode 100644 dump/src/reader/v2/snapshots/dump__reader__v2__test__read_dump_v2_from_meilisearch_v0_22_0_issue_3435-4.snap create mode 100644 dump/src/reader/v2/snapshots/dump__reader__v2__test__read_dump_v2_from_meilisearch_v0_22_0_issue_3435-7.snap create mode 100644 dump/tests/assets/v2-v0.22.0.dump diff --git a/dump/src/reader/compat/snapshots/dump__reader__compat__v1_to_v2__test__compat_v1_v2-3.snap b/dump/src/reader/compat/snapshots/dump__reader__compat__v1_to_v2__test__compat_v1_v2-3.snap index f7e1736b1..8edf789d0 100644 --- a/dump/src/reader/compat/snapshots/dump__reader__compat__v1_to_v2__test__compat_v1_v2-3.snap +++ b/dump/src/reader/compat/snapshots/dump__reader__compat__v1_to_v2__test__compat_v1_v2-3.snap @@ -10,6 +10,7 @@ expression: products.settings().unwrap() "*" ], "filterableAttributes": [], + "sortableAttributes": [], "rankingRules": [ "typo", "words", diff --git a/dump/src/reader/compat/snapshots/dump__reader__compat__v1_to_v2__test__compat_v1_v2-6.snap b/dump/src/reader/compat/snapshots/dump__reader__compat__v1_to_v2__test__compat_v1_v2-6.snap index 8c36fe96c..80c26874b 100644 --- a/dump/src/reader/compat/snapshots/dump__reader__compat__v1_to_v2__test__compat_v1_v2-6.snap +++ b/dump/src/reader/compat/snapshots/dump__reader__compat__v1_to_v2__test__compat_v1_v2-6.snap @@ -13,13 +13,17 @@ expression: movies.settings().unwrap() "genres", "id" ], + "sortableAttributes": [ + "genres", + "id" + ], "rankingRules": [ "typo", "words", "proximity", "attribute", "exactness", - "asc(release_date)" + "release_date:asc" ], "stopWords": [], "synonyms": {}, diff --git a/dump/src/reader/compat/snapshots/dump__reader__compat__v1_to_v2__test__compat_v1_v2-9.snap b/dump/src/reader/compat/snapshots/dump__reader__compat__v1_to_v2__test__compat_v1_v2-9.snap index 1adf85e6a..89da27c25 100644 --- a/dump/src/reader/compat/snapshots/dump__reader__compat__v1_to_v2__test__compat_v1_v2-9.snap +++ b/dump/src/reader/compat/snapshots/dump__reader__compat__v1_to_v2__test__compat_v1_v2-9.snap @@ -10,6 +10,7 @@ expression: spells.settings().unwrap() "*" ], "filterableAttributes": [], + "sortableAttributes": [], "rankingRules": [ "typo", "words", diff --git a/dump/src/reader/compat/v1_to_v2.rs b/dump/src/reader/compat/v1_to_v2.rs index 741d18fa8..baadd2104 100644 --- a/dump/src/reader/compat/v1_to_v2.rs +++ b/dump/src/reader/compat/v1_to_v2.rs @@ -1,4 +1,3 @@ -use std::collections::BTreeSet; use std::str::FromStr; use super::v2_to_v3::CompatV2ToV3; @@ -102,14 +101,15 @@ impl CompatIndexV1ToV2 { impl From for v2::Settings { fn from(source: v1::settings::Settings) -> Self { - let displayed_attributes = source - .displayed_attributes - .map(|opt| opt.map(|displayed_attributes| displayed_attributes.into_iter().collect())); - let attributes_for_faceting = source.attributes_for_faceting.map(|opt| { - opt.map(|attributes_for_faceting| attributes_for_faceting.into_iter().collect()) - }); - let ranking_rules = source.ranking_rules.map(|opt| { - opt.map(|ranking_rules| { + Self { + displayed_attributes: option_to_setting(source.displayed_attributes) + .map(|displayed| displayed.into_iter().collect()), + searchable_attributes: option_to_setting(source.searchable_attributes), + filterable_attributes: option_to_setting(source.attributes_for_faceting.clone()) + .map(|filterable| filterable.into_iter().collect()), + sortable_attributes: option_to_setting(source.attributes_for_faceting) + .map(|sortable| sortable.into_iter().collect()), + ranking_rules: option_to_setting(source.ranking_rules).map(|ranking_rules| { ranking_rules .into_iter() .filter_map(|ranking_rule| { @@ -119,26 +119,33 @@ impl From for v2::Settings { ranking_rule.into(); criterion.as_ref().map(ToString::to_string) } - Err(()) => Some(ranking_rule), + Err(()) => { + log::warn!( + "Could not import the following ranking rule: `{}`.", + ranking_rule + ); + None + } } }) .collect() - }) - }); - - Self { - displayed_attributes, - searchable_attributes: source.searchable_attributes, - filterable_attributes: attributes_for_faceting, - ranking_rules, - stop_words: source.stop_words, - synonyms: source.synonyms, - distinct_attribute: source.distinct_attribute, + }), + stop_words: option_to_setting(source.stop_words), + synonyms: option_to_setting(source.synonyms), + distinct_attribute: option_to_setting(source.distinct_attribute), _kind: std::marker::PhantomData, } } } +fn option_to_setting(opt: Option>) -> v2::Setting { + match opt { + Some(Some(t)) => v2::Setting::Set(t), + None => v2::Setting::NotSet, + Some(None) => v2::Setting::Reset, + } +} + impl From for Option { fn from(source: v1::update::UpdateStatus) -> Self { use v1::update::UpdateStatus as UpdateStatusV1; @@ -251,38 +258,27 @@ impl From for Option { impl From for v2::Settings { fn from(source: v1::settings::SettingsUpdate) -> Self { - let displayed_attributes: Option>> = - source.displayed_attributes.into(); - - let attributes_for_faceting: Option>> = - source.attributes_for_faceting.into(); - - let ranking_rules: Option>> = - source.ranking_rules.into(); + let ranking_rules = v2::Setting::from(source.ranking_rules); // go from the concrete types of v1 (RankingRule) to the concrete type of v2 (Criterion), // and then back to string as this is what the settings manipulate - let ranking_rules = ranking_rules.map(|opt| { - opt.map(|ranking_rules| { - ranking_rules - .into_iter() - // filter out the WordsPosition ranking rule that exists in v1 but not v2 - .filter_map(|ranking_rule| { - Option::::from(ranking_rule) - }) - .map(|criterion| criterion.to_string()) - .collect() - }) + let ranking_rules = ranking_rules.map(|ranking_rules| { + ranking_rules + .into_iter() + // filter out the WordsPosition ranking rule that exists in v1 but not v2 + .filter_map(|ranking_rule| Option::::from(ranking_rule)) + .map(|criterion| criterion.to_string()) + .collect() }); Self { - displayed_attributes: displayed_attributes.map(|opt| { - opt.map(|displayed_attributes| displayed_attributes.into_iter().collect()) - }), + displayed_attributes: v2::Setting::from(source.displayed_attributes) + .map(|displayed_attributes| displayed_attributes.into_iter().collect()), searchable_attributes: source.searchable_attributes.into(), - filterable_attributes: attributes_for_faceting.map(|opt| { - opt.map(|attributes_for_faceting| attributes_for_faceting.into_iter().collect()) - }), + filterable_attributes: v2::Setting::from(source.attributes_for_faceting.clone()) + .map(|attributes_for_faceting| attributes_for_faceting.into_iter().collect()), + sortable_attributes: v2::Setting::from(source.attributes_for_faceting) + .map(|attributes_for_faceting| attributes_for_faceting.into_iter().collect()), ranking_rules, stop_words: source.stop_words.into(), synonyms: source.synonyms.into(), @@ -314,12 +310,12 @@ impl From for Option { } } -impl From> for Option> { +impl From> for v2::Setting { fn from(source: v1::settings::UpdateState) -> Self { match source { - v1::settings::UpdateState::Update(new_value) => Some(Some(new_value)), - v1::settings::UpdateState::Clear => Some(None), - v1::settings::UpdateState::Nothing => None, + v1::settings::UpdateState::Update(new_value) => v2::Setting::Set(new_value), + v1::settings::UpdateState::Clear => v2::Setting::Reset, + v1::settings::UpdateState::Nothing => v2::Setting::NotSet, } } } diff --git a/dump/src/reader/compat/v2_to_v3.rs b/dump/src/reader/compat/v2_to_v3.rs index 8574e04b4..14fc0ee4d 100644 --- a/dump/src/reader/compat/v2_to_v3.rs +++ b/dump/src/reader/compat/v2_to_v3.rs @@ -361,28 +361,29 @@ impl From for v3::Code { } } -fn option_to_setting(opt: Option>) -> v3::Setting { - match opt { - Some(Some(t)) => v3::Setting::Set(t), - None => v3::Setting::NotSet, - Some(None) => v3::Setting::Reset, +impl From> for v3::Setting { + fn from(setting: v2::Setting) -> Self { + match setting { + v2::settings::Setting::Set(a) => v3::settings::Setting::Set(a), + v2::settings::Setting::Reset => v3::settings::Setting::Reset, + v2::settings::Setting::NotSet => v3::settings::Setting::NotSet, + } } } impl From> for v3::Settings { fn from(settings: v2::Settings) -> Self { v3::Settings { - displayed_attributes: option_to_setting(settings.displayed_attributes), - searchable_attributes: option_to_setting(settings.searchable_attributes), - filterable_attributes: option_to_setting(settings.filterable_attributes) - .map(|f| f.into_iter().collect()), - sortable_attributes: v3::Setting::NotSet, - ranking_rules: option_to_setting(settings.ranking_rules).map(|criteria| { + displayed_attributes: settings.displayed_attributes.into(), + searchable_attributes: settings.searchable_attributes.into(), + filterable_attributes: settings.filterable_attributes.into(), + sortable_attributes: settings.sortable_attributes.into(), + ranking_rules: v3::Setting::from(settings.ranking_rules).map(|criteria| { criteria.into_iter().map(|criterion| patch_ranking_rules(&criterion)).collect() }), - stop_words: option_to_setting(settings.stop_words), - synonyms: option_to_setting(settings.synonyms), - distinct_attribute: option_to_setting(settings.distinct_attribute), + stop_words: settings.stop_words.into(), + synonyms: settings.synonyms.into(), + distinct_attribute: settings.distinct_attribute.into(), _kind: std::marker::PhantomData, } } @@ -394,6 +395,7 @@ fn patch_ranking_rules(ranking_rule: &str) -> String { Ok(v2::settings::Criterion::Typo) => String::from("typo"), Ok(v2::settings::Criterion::Proximity) => String::from("proximity"), Ok(v2::settings::Criterion::Attribute) => String::from("attribute"), + Ok(v2::settings::Criterion::Sort) => String::from("sort"), Ok(v2::settings::Criterion::Exactness) => String::from("exactness"), Ok(v2::settings::Criterion::Asc(name)) => format!("{name}:asc"), Ok(v2::settings::Criterion::Desc(name)) => format!("{name}:desc"), diff --git a/dump/src/reader/mod.rs b/dump/src/reader/mod.rs index cf671ea45..a5a66591b 100644 --- a/dump/src/reader/mod.rs +++ b/dump/src/reader/mod.rs @@ -530,6 +530,82 @@ pub(crate) mod test { meili_snap::snapshot_hash!(format!("{:#?}", documents), @"235016433dd04262c7f2da01d1e808ce"); } + #[test] + fn import_dump_v2_from_meilisearch_v0_22_0_issue_3435() { + let dump = File::open("tests/assets/v2-v0.22.0.dump").unwrap(); + let mut dump = DumpReader::open(dump).unwrap(); + + // top level infos + insta::assert_display_snapshot!(dump.date().unwrap(), @"2023-01-30 16:26:09.247261 +00:00:00"); + assert_eq!(dump.instance_uid().unwrap(), None); + + // tasks + let tasks = dump.tasks().unwrap().collect::>>().unwrap(); + let (tasks, update_files): (Vec<_>, Vec<_>) = tasks.into_iter().unzip(); + meili_snap::snapshot_hash!(meili_snap::json_string!(tasks), @"2db37756d8af1fb7623436b76e8956a6"); + assert_eq!(update_files.len(), 8); + assert!(update_files[0..].iter().all(|u| u.is_none())); // everything already processed + + // keys + let keys = dump.keys().unwrap().collect::>>().unwrap(); + meili_snap::snapshot_hash!(meili_snap::json_string!(keys), @"d751713988987e9331980363e24189ce"); + + // indexes + let mut indexes = dump.indexes().unwrap().collect::>>().unwrap(); + // the index are not ordered in any way by default + indexes.sort_by_key(|index| index.metadata().uid.to_string()); + + let mut products = indexes.pop().unwrap(); + let mut movies = indexes.pop().unwrap(); + let mut spells = indexes.pop().unwrap(); + assert!(indexes.is_empty()); + + // products + insta::assert_json_snapshot!(products.metadata(), { ".createdAt" => "[now]", ".updatedAt" => "[now]" }, @r###" + { + "uid": "products", + "primaryKey": "sku", + "createdAt": "[now]", + "updatedAt": "[now]" + } + "###); + + insta::assert_json_snapshot!(products.settings().unwrap()); + let documents = products.documents().unwrap().collect::>>().unwrap(); + assert_eq!(documents.len(), 10); + meili_snap::snapshot_hash!(format!("{:#?}", documents), @"548284a84de510f71e88e6cdea495cf5"); + + // movies + insta::assert_json_snapshot!(movies.metadata(), { ".createdAt" => "[now]", ".updatedAt" => "[now]" }, @r###" + { + "uid": "movies", + "primaryKey": "id", + "createdAt": "[now]", + "updatedAt": "[now]" + } + "###); + + insta::assert_json_snapshot!(movies.settings().unwrap()); + let documents = movies.documents().unwrap().collect::>>().unwrap(); + assert_eq!(documents.len(), 10); + meili_snap::snapshot_hash!(format!("{:#?}", documents), @"0227598af846e574139ee0b80e03a720"); + + // spells + insta::assert_json_snapshot!(spells.metadata(), { ".createdAt" => "[now]", ".updatedAt" => "[now]" }, @r###" + { + "uid": "dnd_spells", + "primaryKey": "index", + "createdAt": "[now]", + "updatedAt": "[now]" + } + "###); + + insta::assert_json_snapshot!(spells.settings().unwrap()); + let documents = spells.documents().unwrap().collect::>>().unwrap(); + assert_eq!(documents.len(), 10); + meili_snap::snapshot_hash!(format!("{:#?}", documents), @"235016433dd04262c7f2da01d1e808ce"); + } + #[test] fn import_dump_v1() { let dump = File::open("tests/assets/v1.dump").unwrap(); @@ -542,7 +618,7 @@ pub(crate) mod test { // tasks let tasks = dump.tasks().unwrap().collect::>>().unwrap(); let (tasks, update_files): (Vec<_>, Vec<_>) = tasks.into_iter().unzip(); - meili_snap::snapshot_hash!(meili_snap::json_string!(tasks), @"b3e3652bfc10a76670be157d2507d761"); + meili_snap::snapshot_hash!(meili_snap::json_string!(tasks), @"8df6eab075a44b3c1af6b726f9fd9a43"); assert_eq!(update_files.len(), 9); assert!(update_files[..].iter().all(|u| u.is_none())); // no update file in dump v1 diff --git a/dump/src/reader/snapshots/dump__reader__test__import_dump_v1-11.snap b/dump/src/reader/snapshots/dump__reader__test__import_dump_v1-11.snap index 997d303e7..92fc61d72 100644 --- a/dump/src/reader/snapshots/dump__reader__test__import_dump_v1-11.snap +++ b/dump/src/reader/snapshots/dump__reader__test__import_dump_v1-11.snap @@ -10,6 +10,7 @@ expression: spells.settings().unwrap() "*" ], "filterableAttributes": [], + "sortableAttributes": [], "rankingRules": [ "typo", "words", diff --git a/dump/src/reader/snapshots/dump__reader__test__import_dump_v1-5.snap b/dump/src/reader/snapshots/dump__reader__test__import_dump_v1-5.snap index 282cd6ba7..b0b54c136 100644 --- a/dump/src/reader/snapshots/dump__reader__test__import_dump_v1-5.snap +++ b/dump/src/reader/snapshots/dump__reader__test__import_dump_v1-5.snap @@ -10,6 +10,7 @@ expression: products.settings().unwrap() "*" ], "filterableAttributes": [], + "sortableAttributes": [], "rankingRules": [ "typo", "words", diff --git a/dump/src/reader/snapshots/dump__reader__test__import_dump_v1-8.snap b/dump/src/reader/snapshots/dump__reader__test__import_dump_v1-8.snap index d20fdc77e..5c12a0438 100644 --- a/dump/src/reader/snapshots/dump__reader__test__import_dump_v1-8.snap +++ b/dump/src/reader/snapshots/dump__reader__test__import_dump_v1-8.snap @@ -13,6 +13,10 @@ expression: movies.settings().unwrap() "genres", "id" ], + "sortableAttributes": [ + "genres", + "id" + ], "rankingRules": [ "typo", "words", diff --git a/dump/src/reader/snapshots/dump__reader__test__import_dump_v2_from_meilisearch_v0_22_0_issue_3435-11.snap b/dump/src/reader/snapshots/dump__reader__test__import_dump_v2_from_meilisearch_v0_22_0_issue_3435-11.snap new file mode 100644 index 000000000..d8a5bafbe --- /dev/null +++ b/dump/src/reader/snapshots/dump__reader__test__import_dump_v2_from_meilisearch_v0_22_0_issue_3435-11.snap @@ -0,0 +1,25 @@ +--- +source: dump/src/reader/mod.rs +expression: spells.settings().unwrap() +--- +{ + "displayedAttributes": [ + "*" + ], + "searchableAttributes": [ + "*" + ], + "filterableAttributes": [], + "sortableAttributes": [], + "rankingRules": [ + "words", + "typo", + "proximity", + "attribute", + "sort", + "exactness" + ], + "stopWords": [], + "synonyms": {}, + "distinctAttribute": null +} diff --git a/dump/src/reader/snapshots/dump__reader__test__import_dump_v2_from_meilisearch_v0_22_0_issue_3435-5.snap b/dump/src/reader/snapshots/dump__reader__test__import_dump_v2_from_meilisearch_v0_22_0_issue_3435-5.snap new file mode 100644 index 000000000..abf97a8ab --- /dev/null +++ b/dump/src/reader/snapshots/dump__reader__test__import_dump_v2_from_meilisearch_v0_22_0_issue_3435-5.snap @@ -0,0 +1,39 @@ +--- +source: dump/src/reader/mod.rs +expression: products.settings().unwrap() +--- +{ + "displayedAttributes": [ + "*" + ], + "searchableAttributes": [ + "*" + ], + "filterableAttributes": [], + "sortableAttributes": [], + "rankingRules": [ + "words", + "typo", + "proximity", + "attribute", + "sort", + "exactness" + ], + "stopWords": [], + "synonyms": { + "android": [ + "phone", + "smartphone" + ], + "iphone": [ + "phone", + "smartphone" + ], + "phone": [ + "android", + "iphone", + "smartphone" + ] + }, + "distinctAttribute": null +} diff --git a/dump/src/reader/snapshots/dump__reader__test__import_dump_v2_from_meilisearch_v0_22_0_issue_3435-8.snap b/dump/src/reader/snapshots/dump__reader__test__import_dump_v2_from_meilisearch_v0_22_0_issue_3435-8.snap new file mode 100644 index 000000000..f02a3685e --- /dev/null +++ b/dump/src/reader/snapshots/dump__reader__test__import_dump_v2_from_meilisearch_v0_22_0_issue_3435-8.snap @@ -0,0 +1,30 @@ +--- +source: dump/src/reader/mod.rs +expression: movies.settings().unwrap() +--- +{ + "displayedAttributes": [ + "*" + ], + "searchableAttributes": [ + "*" + ], + "filterableAttributes": [ + "genres", + "id" + ], + "sortableAttributes": [ + "release_date" + ], + "rankingRules": [ + "words", + "typo", + "proximity", + "attribute", + "exactness", + "release_date:asc" + ], + "stopWords": [], + "synonyms": {}, + "distinctAttribute": null +} diff --git a/dump/src/reader/v2/mod.rs b/dump/src/reader/v2/mod.rs index befebbdb3..4016e6341 100644 --- a/dump/src/reader/v2/mod.rs +++ b/dump/src/reader/v2/mod.rs @@ -41,6 +41,7 @@ use super::Document; use crate::{IndexMetadata, Result, Version}; pub type Settings = settings::Settings; +pub type Setting = settings::Setting; pub type Checked = settings::Checked; pub type Unchecked = settings::Unchecked; @@ -306,4 +307,81 @@ pub(crate) mod test { assert_eq!(documents.len(), 10); meili_snap::snapshot_hash!(format!("{:#?}", documents), @"235016433dd04262c7f2da01d1e808ce"); } + + #[test] + fn read_dump_v2_from_meilisearch_v0_22_0_issue_3435() { + let dump = File::open("tests/assets/v2-v0.22.0.dump").unwrap(); + let dir = TempDir::new().unwrap(); + let mut dump = BufReader::new(dump); + let gz = GzDecoder::new(&mut dump); + let mut archive = tar::Archive::new(gz); + archive.unpack(dir.path()).unwrap(); + + let mut dump = V2Reader::open(dir).unwrap(); + + // top level infos + insta::assert_display_snapshot!(dump.date().unwrap(), @"2023-01-30 16:26:09.247261 +00:00:00"); + + // tasks + let tasks = dump.tasks().collect::>>().unwrap(); + let (tasks, update_files): (Vec<_>, Vec<_>) = tasks.into_iter().unzip(); + meili_snap::snapshot_hash!(meili_snap::json_string!(tasks), @"aca8ba13046272664eb3ea2da3031633"); + assert_eq!(update_files.len(), 8); + assert!(update_files[0..].iter().all(|u| u.is_none())); // everything has already been processed + + // indexes + let mut indexes = dump.indexes().unwrap().collect::>>().unwrap(); + // the index are not ordered in any way by default + indexes.sort_by_key(|index| index.metadata().uid.to_string()); + + let mut products = indexes.pop().unwrap(); + let mut movies = indexes.pop().unwrap(); + let mut spells = indexes.pop().unwrap(); + assert!(indexes.is_empty()); + + // products + insta::assert_json_snapshot!(products.metadata(), { ".createdAt" => "[now]", ".updatedAt" => "[now]" }, @r###" + { + "uid": "products", + "primaryKey": "sku", + "createdAt": "[now]", + "updatedAt": "[now]" + } + "###); + + insta::assert_json_snapshot!(products.settings().unwrap()); + let documents = products.documents().unwrap().collect::>>().unwrap(); + assert_eq!(documents.len(), 10); + meili_snap::snapshot_hash!(format!("{:#?}", documents), @"548284a84de510f71e88e6cdea495cf5"); + + // movies + insta::assert_json_snapshot!(movies.metadata(), { ".createdAt" => "[now]", ".updatedAt" => "[now]" }, @r###" + { + "uid": "movies", + "primaryKey": "id", + "createdAt": "[now]", + "updatedAt": "[now]" + } + "###); + + insta::assert_json_snapshot!(movies.settings().unwrap()); + let documents = movies.documents().unwrap().collect::>>().unwrap(); + assert_eq!(documents.len(), 10); + meili_snap::snapshot_hash!(format!("{:#?}", documents), @"0227598af846e574139ee0b80e03a720"); + + // spells + insta::assert_json_snapshot!(spells.metadata(), { ".createdAt" => "[now]", ".updatedAt" => "[now]" }, @r###" + { + "uid": "dnd_spells", + "primaryKey": "index", + "createdAt": "[now]", + "updatedAt": "[now]" + } + "###); + + insta::assert_json_snapshot!(spells.settings().unwrap()); + let documents = spells.documents().unwrap().collect::>>().unwrap(); + assert_eq!(documents.len(), 10); + meili_snap::snapshot_hash!(format!("{:#?}", documents), @"235016433dd04262c7f2da01d1e808ce"); + } } diff --git a/dump/src/reader/v2/settings.rs b/dump/src/reader/v2/settings.rs index 1a7935b56..9cd363ca5 100644 --- a/dump/src/reader/v2/settings.rs +++ b/dump/src/reader/v2/settings.rs @@ -1,35 +1,33 @@ use std::collections::{BTreeMap, BTreeSet}; -use std::fmt::Display; +use std::fmt; use std::marker::PhantomData; use std::str::FromStr; -use once_cell::sync::Lazy; -use regex::Regex; use serde::{Deserialize, Deserializer}; #[cfg(test)] fn serialize_with_wildcard( - field: &Option>>, + field: &Setting>, s: S, ) -> std::result::Result where S: serde::Serializer, { - let wildcard = vec!["*".to_string()]; - s.serialize_some(&field.as_ref().map(|o| o.as_ref().unwrap_or(&wildcard))) -} + use serde::Serialize; -fn deserialize_some<'de, T, D>(deserializer: D) -> std::result::Result, D::Error> -where - T: Deserialize<'de>, - D: Deserializer<'de>, -{ - Deserialize::deserialize(deserializer).map(Some) + let wildcard = vec!["*".to_string()]; + match field { + Setting::Set(value) => Some(value), + Setting::Reset => Some(&wildcard), + Setting::NotSet => None, + } + .serialize(s) } #[derive(Clone, Default, Debug)] #[cfg_attr(test, derive(serde::Serialize))] pub struct Checked; + #[derive(Clone, Default, Debug, Deserialize)] #[cfg_attr(test, derive(serde::Serialize))] pub struct Unchecked; @@ -42,75 +40,54 @@ pub struct Unchecked; pub struct Settings { #[serde( default, - deserialize_with = "deserialize_some", serialize_with = "serialize_with_wildcard", - skip_serializing_if = "Option::is_none" + skip_serializing_if = "Setting::is_not_set" )] - pub displayed_attributes: Option>>, + pub displayed_attributes: Setting>, #[serde( default, - deserialize_with = "deserialize_some", serialize_with = "serialize_with_wildcard", - skip_serializing_if = "Option::is_none" + skip_serializing_if = "Setting::is_not_set" )] - pub searchable_attributes: Option>>, + pub searchable_attributes: Setting>, - #[serde( - default, - deserialize_with = "deserialize_some", - skip_serializing_if = "Option::is_none" - )] - pub filterable_attributes: Option>>, - - #[serde( - default, - deserialize_with = "deserialize_some", - skip_serializing_if = "Option::is_none" - )] - pub ranking_rules: Option>>, - #[serde( - default, - deserialize_with = "deserialize_some", - skip_serializing_if = "Option::is_none" - )] - pub stop_words: Option>>, - #[serde( - default, - deserialize_with = "deserialize_some", - skip_serializing_if = "Option::is_none" - )] - pub synonyms: Option>>>, - #[serde( - default, - deserialize_with = "deserialize_some", - skip_serializing_if = "Option::is_none" - )] - pub distinct_attribute: Option>, + #[serde(default, skip_serializing_if = "Setting::is_not_set")] + pub filterable_attributes: Setting>, + #[serde(default, skip_serializing_if = "Setting::is_not_set")] + pub sortable_attributes: Setting>, + #[serde(default, skip_serializing_if = "Setting::is_not_set")] + pub ranking_rules: Setting>, + #[serde(default, skip_serializing_if = "Setting::is_not_set")] + pub stop_words: Setting>, + #[serde(default, skip_serializing_if = "Setting::is_not_set")] + pub synonyms: Setting>>, + #[serde(default, skip_serializing_if = "Setting::is_not_set")] + pub distinct_attribute: Setting, #[serde(skip)] pub _kind: PhantomData, } impl Settings { - pub fn check(mut self) -> Settings { - let displayed_attributes = match self.displayed_attributes.take() { - Some(Some(fields)) => { + pub fn check(self) -> Settings { + let displayed_attributes = match self.displayed_attributes { + Setting::Set(fields) => { if fields.iter().any(|f| f == "*") { - Some(None) + Setting::Reset } else { - Some(Some(fields)) + Setting::Set(fields) } } otherwise => otherwise, }; - let searchable_attributes = match self.searchable_attributes.take() { - Some(Some(fields)) => { + let searchable_attributes = match self.searchable_attributes { + Setting::Set(fields) => { if fields.iter().any(|f| f == "*") { - Some(None) + Setting::Reset } else { - Some(Some(fields)) + Setting::Set(fields) } } otherwise => otherwise, @@ -120,6 +97,7 @@ impl Settings { displayed_attributes, searchable_attributes, filterable_attributes: self.filterable_attributes, + sortable_attributes: self.sortable_attributes, ranking_rules: self.ranking_rules, stop_words: self.stop_words, synonyms: self.synonyms, @@ -129,10 +107,61 @@ impl Settings { } } -static ASC_DESC_REGEX: Lazy = - Lazy::new(|| Regex::new(r#"(asc|desc)\(([\w_-]+)\)"#).unwrap()); +#[derive(Debug, Clone, PartialEq)] +pub enum Setting { + Set(T), + Reset, + NotSet, +} -#[derive(Debug, Deserialize, Clone, PartialEq, Eq)] +impl Default for Setting { + fn default() -> Self { + Self::NotSet + } +} + +impl Setting { + pub const fn is_not_set(&self) -> bool { + matches!(self, Self::NotSet) + } + + pub fn map(self, f: fn(T) -> A) -> Setting { + match self { + Setting::Set(a) => Setting::Set(f(a)), + Setting::Reset => Setting::Reset, + Setting::NotSet => Setting::NotSet, + } + } +} + +#[cfg(test)] +impl serde::Serialize for Setting { + fn serialize(&self, serializer: S) -> std::result::Result + where + S: serde::Serializer, + { + match self { + Self::Set(value) => Some(value), + // Usually not_set isn't serialized by setting skip_serializing_if field attribute + Self::NotSet | Self::Reset => None, + } + .serialize(serializer) + } +} + +impl<'de, T: Deserialize<'de>> Deserialize<'de> for Setting { + fn deserialize(deserializer: D) -> std::result::Result + where + D: Deserializer<'de>, + { + Deserialize::deserialize(deserializer).map(|x| match x { + Some(x) => Self::Set(x), + None => Self::Reset, // Reset is forced by sending null value + }) + } +} + +#[derive(Debug, Clone, PartialEq, Eq)] pub enum Criterion { /// Sorted by decreasing number of matched query terms. /// Query words at the front of an attribute is considered better than if it was at the back. @@ -142,8 +171,11 @@ pub enum Criterion { /// Sorted by increasing distance between matched query terms. Proximity, /// Documents with quey words contained in more important - /// attributes are considred better. + /// attributes are considered better. Attribute, + /// Dynamically sort at query time the documents. None, one or multiple Asc/Desc sortable + /// attributes can be used in place of this criterion at query time. + Sort, /// Sorted by the similarity of the matched words with the query words. Exactness, /// Sorted by the increasing value of the field specified. @@ -152,40 +184,86 @@ pub enum Criterion { Desc(String), } +impl Criterion { + /// Returns the field name parameter of this criterion. + pub fn field_name(&self) -> Option<&str> { + match self { + Criterion::Asc(name) | Criterion::Desc(name) => Some(name), + _otherwise => None, + } + } +} + impl FromStr for Criterion { + // since we're not going to show the custom error message we can override the + // error type. type Err = (); - fn from_str(txt: &str) -> Result { - match txt { + fn from_str(text: &str) -> Result { + match text { "words" => Ok(Criterion::Words), "typo" => Ok(Criterion::Typo), "proximity" => Ok(Criterion::Proximity), "attribute" => Ok(Criterion::Attribute), + "sort" => Ok(Criterion::Sort), "exactness" => Ok(Criterion::Exactness), - text => { - let caps = ASC_DESC_REGEX.captures(text).ok_or(())?; - let order = caps.get(1).unwrap().as_str(); - let field_name = caps.get(2).unwrap().as_str(); - match order { - "asc" => Ok(Criterion::Asc(field_name.to_string())), - "desc" => Ok(Criterion::Desc(field_name.to_string())), - _text => Err(()), - } - } + text => match AscDesc::from_str(text) { + Ok(AscDesc::Asc(field)) => Ok(Criterion::Asc(field)), + Ok(AscDesc::Desc(field)) => Ok(Criterion::Desc(field)), + Err(_) => Err(()), + }, } } } -impl Display for Criterion { - fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { - match self { - Criterion::Words => write!(f, "words"), - Criterion::Typo => write!(f, "typo"), - Criterion::Proximity => write!(f, "proximity"), - Criterion::Attribute => write!(f, "attribute"), - Criterion::Exactness => write!(f, "exactness"), - Criterion::Asc(field_name) => write!(f, "asc({})", field_name), - Criterion::Desc(field_name) => write!(f, "desc({})", field_name), +#[derive(Debug, Deserialize, Clone, PartialEq, Eq)] +pub enum AscDesc { + Asc(String), + Desc(String), +} + +impl FromStr for AscDesc { + type Err = (); + + // since we don't know if this comes from the old or new syntax we need to check + // for both syntax. + // WARN: this code doesn't come from the original meilisearch v0.22.0 but was + // written specifically to be able to import the dump of meilisearch v0.21.0 AND + // meilisearch v0.22.0. + fn from_str(text: &str) -> Result { + if let Some((field_name, asc_desc)) = text.rsplit_once(':') { + match asc_desc { + "asc" => Ok(AscDesc::Asc(field_name.to_string())), + "desc" => Ok(AscDesc::Desc(field_name.to_string())), + _ => Err(()), + } + } else if text.starts_with("asc(") && text.ends_with(")") { + Ok(AscDesc::Asc( + text.strip_prefix("asc(").unwrap().strip_suffix(")").unwrap().to_string(), + )) + } else if text.starts_with("desc(") && text.ends_with(")") { + Ok(AscDesc::Desc( + text.strip_prefix("desc(").unwrap().strip_suffix(")").unwrap().to_string(), + )) + } else { + Err(()) + } + } +} + +impl fmt::Display for Criterion { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + use Criterion::*; + + match self { + Words => f.write_str("words"), + Typo => f.write_str("typo"), + Proximity => f.write_str("proximity"), + Attribute => f.write_str("attribute"), + Sort => f.write_str("sort"), + Exactness => f.write_str("exactness"), + Asc(attr) => write!(f, "{}:asc", attr), + Desc(attr) => write!(f, "{}:desc", attr), } } } diff --git a/dump/src/reader/v2/snapshots/dump__reader__v2__test__read_dump_v2_from_meilisearch_v0_22_0_issue_3435-10.snap b/dump/src/reader/v2/snapshots/dump__reader__v2__test__read_dump_v2_from_meilisearch_v0_22_0_issue_3435-10.snap new file mode 100644 index 000000000..fbff0a0e6 --- /dev/null +++ b/dump/src/reader/v2/snapshots/dump__reader__v2__test__read_dump_v2_from_meilisearch_v0_22_0_issue_3435-10.snap @@ -0,0 +1,25 @@ +--- +source: dump/src/reader/v2/mod.rs +expression: spells.settings().unwrap() +--- +{ + "displayedAttributes": [ + "*" + ], + "searchableAttributes": [ + "*" + ], + "filterableAttributes": [], + "sortableAttributes": [], + "rankingRules": [ + "words", + "typo", + "proximity", + "attribute", + "sort", + "exactness" + ], + "stopWords": [], + "synonyms": {}, + "distinctAttribute": null +} diff --git a/dump/src/reader/v2/snapshots/dump__reader__v2__test__read_dump_v2_from_meilisearch_v0_22_0_issue_3435-4.snap b/dump/src/reader/v2/snapshots/dump__reader__v2__test__read_dump_v2_from_meilisearch_v0_22_0_issue_3435-4.snap new file mode 100644 index 000000000..809ced4f6 --- /dev/null +++ b/dump/src/reader/v2/snapshots/dump__reader__v2__test__read_dump_v2_from_meilisearch_v0_22_0_issue_3435-4.snap @@ -0,0 +1,39 @@ +--- +source: dump/src/reader/v2/mod.rs +expression: products.settings().unwrap() +--- +{ + "displayedAttributes": [ + "*" + ], + "searchableAttributes": [ + "*" + ], + "filterableAttributes": [], + "sortableAttributes": [], + "rankingRules": [ + "words", + "typo", + "proximity", + "attribute", + "sort", + "exactness" + ], + "stopWords": [], + "synonyms": { + "android": [ + "phone", + "smartphone" + ], + "iphone": [ + "phone", + "smartphone" + ], + "phone": [ + "android", + "iphone", + "smartphone" + ] + }, + "distinctAttribute": null +} diff --git a/dump/src/reader/v2/snapshots/dump__reader__v2__test__read_dump_v2_from_meilisearch_v0_22_0_issue_3435-7.snap b/dump/src/reader/v2/snapshots/dump__reader__v2__test__read_dump_v2_from_meilisearch_v0_22_0_issue_3435-7.snap new file mode 100644 index 000000000..016ada2a3 --- /dev/null +++ b/dump/src/reader/v2/snapshots/dump__reader__v2__test__read_dump_v2_from_meilisearch_v0_22_0_issue_3435-7.snap @@ -0,0 +1,30 @@ +--- +source: dump/src/reader/v2/mod.rs +expression: movies.settings().unwrap() +--- +{ + "displayedAttributes": [ + "*" + ], + "searchableAttributes": [ + "*" + ], + "filterableAttributes": [ + "genres", + "id" + ], + "sortableAttributes": [ + "release_date" + ], + "rankingRules": [ + "words", + "typo", + "proximity", + "attribute", + "exactness", + "release_date:asc" + ], + "stopWords": [], + "synonyms": {}, + "distinctAttribute": null +} diff --git a/dump/tests/assets/v2-v0.22.0.dump b/dump/tests/assets/v2-v0.22.0.dump new file mode 100644 index 0000000000000000000000000000000000000000..8932284eaa591dd569b0ed277594796250afda9c GIT binary patch literal 9809 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 literal 0 HcmV?d00001 diff --git a/meilisearch/tests/dumps/mod.rs b/meilisearch/tests/dumps/mod.rs index 06cb21f20..cbc895f32 100644 --- a/meilisearch/tests/dumps/mod.rs +++ b/meilisearch/tests/dumps/mod.rs @@ -98,14 +98,14 @@ async fn import_dump_v1_movie_with_settings() { assert_eq!(code, 200); assert_eq!( settings, - json!({ "displayedAttributes": ["genres", "id", "overview", "poster", "release_date", "title"], "searchableAttributes": ["title", "overview"], "filterableAttributes": ["genres"], "sortableAttributes": [], "rankingRules": ["typo", "words", "proximity", "attribute", "exactness"], "stopWords": ["of", "the"], "synonyms": {}, "distinctAttribute": null, "typoTolerance": {"enabled": true, "minWordSizeForTypos": { "oneTypo": 5, "twoTypos": 9 }, "disableOnWords": [], "disableOnAttributes": [] }, "faceting": { "maxValuesPerFacet": 100 }, "pagination": { "maxTotalHits": 1000 } }) + json!({ "displayedAttributes": ["genres", "id", "overview", "poster", "release_date", "title"], "searchableAttributes": ["title", "overview"], "filterableAttributes": ["genres"], "sortableAttributes": ["genres"], "rankingRules": ["typo", "words", "proximity", "attribute", "exactness"], "stopWords": ["of", "the"], "synonyms": {}, "distinctAttribute": null, "typoTolerance": {"enabled": true, "minWordSizeForTypos": { "oneTypo": 5, "twoTypos": 9 }, "disableOnWords": [], "disableOnAttributes": [] }, "faceting": { "maxValuesPerFacet": 100 }, "pagination": { "maxTotalHits": 1000 } }) ); let (tasks, code) = index.list_tasks().await; assert_eq!(code, 200); assert_eq!( tasks, - json!({ "results": [{ "uid": 1, "indexUid": "indexUID", "status": "succeeded", "type": "settingsUpdate", "canceledBy": null, "details": { "displayedAttributes": ["genres", "id", "overview", "poster", "release_date", "title"], "searchableAttributes": ["title", "overview"], "filterableAttributes": ["genres"], "stopWords": ["of", "the"] }, "error": null, "duration": "PT7.288826907S", "enqueuedAt": "2021-09-08T09:34:40.882977Z", "startedAt": "2021-09-08T09:34:40.883073093Z", "finishedAt": "2021-09-08T09:34:48.1719Z"}, { "uid": 0, "indexUid": "indexUID", "status": "succeeded", "type": "documentAdditionOrUpdate", "canceledBy": null, "details": { "receivedDocuments": 0, "indexedDocuments": 31968 }, "error": null, "duration": "PT9.090735774S", "enqueuedAt": "2021-09-08T09:34:16.036101Z", "startedAt": "2021-09-08T09:34:16.261191226Z", "finishedAt": "2021-09-08T09:34:25.351927Z" }], "limit": 20, "from": 1, "next": null }) + json!({ "results": [{ "uid": 1, "indexUid": "indexUID", "status": "succeeded", "type": "settingsUpdate", "canceledBy": null, "details": { "displayedAttributes": ["genres", "id", "overview", "poster", "release_date", "title"], "searchableAttributes": ["title", "overview"], "filterableAttributes": ["genres"], "sortableAttributes": ["genres"], "stopWords": ["of", "the"] }, "error": null, "duration": "PT7.288826907S", "enqueuedAt": "2021-09-08T09:34:40.882977Z", "startedAt": "2021-09-08T09:34:40.883073093Z", "finishedAt": "2021-09-08T09:34:48.1719Z"}, { "uid": 0, "indexUid": "indexUID", "status": "succeeded", "type": "documentAdditionOrUpdate", "canceledBy": null, "details": { "receivedDocuments": 0, "indexedDocuments": 31968 }, "error": null, "duration": "PT9.090735774S", "enqueuedAt": "2021-09-08T09:34:16.036101Z", "startedAt": "2021-09-08T09:34:16.261191226Z", "finishedAt": "2021-09-08T09:34:25.351927Z" }], "limit": 20, "from": 1, "next": null }) ); // finally we're just going to check that we can still get a few documents by id @@ -161,7 +161,7 @@ async fn import_dump_v1_rubygems_with_settings() { assert_eq!(code, 200); assert_eq!( settings, - json!({"displayedAttributes": ["description", "id", "name", "summary", "total_downloads", "version"], "searchableAttributes": ["name", "summary"], "filterableAttributes": ["version"], "sortableAttributes": [], "rankingRules": ["typo", "words", "fame:desc", "proximity", "attribute", "exactness", "total_downloads:desc"], "stopWords": [], "synonyms": {}, "distinctAttribute": null, "typoTolerance": {"enabled": true, "minWordSizeForTypos": {"oneTypo": 5, "twoTypos": 9}, "disableOnWords": [], "disableOnAttributes": [] }, "faceting": { "maxValuesPerFacet": 100 }, "pagination": { "maxTotalHits": 1000 }}) + json!({"displayedAttributes": ["description", "id", "name", "summary", "total_downloads", "version"], "searchableAttributes": ["name", "summary"], "filterableAttributes": ["version"], "sortableAttributes": ["version"], "rankingRules": ["typo", "words", "fame:desc", "proximity", "attribute", "exactness", "total_downloads:desc"], "stopWords": [], "synonyms": {}, "distinctAttribute": null, "typoTolerance": {"enabled": true, "minWordSizeForTypos": {"oneTypo": 5, "twoTypos": 9}, "disableOnWords": [], "disableOnAttributes": [] }, "faceting": { "maxValuesPerFacet": 100 }, "pagination": { "maxTotalHits": 1000 }}) ); let (tasks, code) = index.list_tasks().await;