mirror of
				https://github.com/meilisearch/meilisearch.git
				synced 2025-10-25 13:06:27 +00:00 
			
		
		
		
	Merge #5254
5254: Granular Filterable attribute settings r=ManyTheFish a=ManyTheFish # Related **Issue:** https://github.com/meilisearch/meilisearch/issues/5163 **PRD:** https://meilisearch.notion.site/API-usage-Settings-to-opt-out-indexing-features-filterableAttributes-1764b06b651f80aba8bdf359b2df3ca8 # Summary Change the `filterableAttributes` settings to let the user choose which facet feature he wants to activate or not. Deactivating a feature will avoid some database computation in the indexing process and save time and disk size. # Example `PATCH /indexes/:index_uid/settings` ```json { "filterableAttributes": [ { "patterns": [ "cattos", "doggos.age" ], "features": { "facetSearch": false, "filter": { "equality": true, "comparison": false } } } ] } ``` # Impact on the codebase - Settings API: - `/settings` - `/settings/filterable-attributes` - OpenAPI - may impact the LocalizedAttributesRules due to the AttributePatterns factorization - Database: - Filterable attributes format changed - Faceted field_ids are no more stored in the database - FieldIdsMap has no more unexisting fields - Search: - Search using filters - Facet search - `Attributes` ranking rule - Distinct attribute - Facet distribution - Settings reindexing: - searchable - facet - vector - geo - Document indexing: - searchable - facet - vector - geo - Dump import # Note for the reviewers The changes are huge and have been split in different commits with a dedicated explanation, I suggest reviewing the commit 1by1 Co-authored-by: ManyTheFish <many@meilisearch.com>
This commit is contained in:
		| @@ -291,7 +291,7 @@ make_setting_routes!( | ||||
|     { | ||||
|         route: "/filterable-attributes", | ||||
|         update_verb: put, | ||||
|         value_type: std::collections::BTreeSet<String>, | ||||
|         value_type: Vec<meilisearch_types::milli::FilterableAttributesRule>, | ||||
|         err_type: meilisearch_types::deserr::DeserrJsonError< | ||||
|             meilisearch_types::error::deserr_codes::InvalidSettingsFilterableAttributes, | ||||
|         >, | ||||
|   | ||||
| @@ -8,6 +8,7 @@ use std::collections::{BTreeMap, BTreeSet, HashSet}; | ||||
| use meilisearch_types::facet_values_sort::FacetValuesSort; | ||||
| use meilisearch_types::locales::{Locale, LocalizedAttributesRuleView}; | ||||
| use meilisearch_types::milli::update::Setting; | ||||
| use meilisearch_types::milli::FilterableAttributesRule; | ||||
| use meilisearch_types::settings::{ | ||||
|     FacetingSettings, PaginationSettings, PrefixSearchSettings, ProximityPrecisionView, | ||||
|     RankingRuleView, SettingEmbeddingSettings, TypoSettings, | ||||
| @@ -89,6 +90,10 @@ impl Aggregate for SettingsAnalytics { | ||||
|             filterable_attributes: FilterableAttributesAnalytics { | ||||
|                 total: new.filterable_attributes.total.or(self.filterable_attributes.total), | ||||
|                 has_geo: new.filterable_attributes.has_geo.or(self.filterable_attributes.has_geo), | ||||
|                 has_patterns: new | ||||
|                     .filterable_attributes | ||||
|                     .has_patterns | ||||
|                     .or(self.filterable_attributes.has_patterns), | ||||
|             }, | ||||
|             distinct_attribute: DistinctAttributeAnalytics { | ||||
|                 set: self.distinct_attribute.set | new.distinct_attribute.set, | ||||
| @@ -328,13 +333,19 @@ impl SortableAttributesAnalytics { | ||||
| pub struct FilterableAttributesAnalytics { | ||||
|     pub total: Option<usize>, | ||||
|     pub has_geo: Option<bool>, | ||||
|     pub has_patterns: Option<bool>, | ||||
| } | ||||
|  | ||||
| impl FilterableAttributesAnalytics { | ||||
|     pub fn new(setting: Option<&BTreeSet<String>>) -> Self { | ||||
|     pub fn new(setting: Option<&Vec<FilterableAttributesRule>>) -> Self { | ||||
|         Self { | ||||
|             total: setting.as_ref().map(|filter| filter.len()), | ||||
|             has_geo: setting.as_ref().map(|filter| filter.contains("_geo")), | ||||
|             has_geo: setting | ||||
|                 .as_ref() | ||||
|                 .map(|filter| filter.iter().any(FilterableAttributesRule::has_geo)), | ||||
|             has_patterns: setting.as_ref().map(|filter| { | ||||
|                 filter.iter().any(|rule| matches!(rule, FilterableAttributesRule::Pattern(_))) | ||||
|             }), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -9,6 +9,10 @@ use meilisearch_types::batches::BatchStats; | ||||
| use meilisearch_types::error::{Code, ErrorType, ResponseError}; | ||||
| use meilisearch_types::index_uid::IndexUid; | ||||
| use meilisearch_types::keys::CreateApiKey; | ||||
| use meilisearch_types::milli::{ | ||||
|     AttributePatterns, FilterFeatures, FilterableAttributesFeatures, FilterableAttributesPatterns, | ||||
|     FilterableAttributesRule, | ||||
| }; | ||||
| use meilisearch_types::settings::{ | ||||
|     Checked, FacetingSettings, MinWordSizeTyposSetting, PaginationSettings, Settings, TypoSettings, | ||||
|     Unchecked, | ||||
| @@ -88,7 +92,7 @@ pub mod tasks; | ||||
|         url = "/", | ||||
|         description = "Local server", | ||||
|     )), | ||||
|     components(schemas(PaginationView<KeyView>, PaginationView<IndexView>, IndexView, DocumentDeletionByFilter, AllBatches, BatchStats, ProgressStepView, ProgressView, BatchView, RuntimeTogglableFeatures, SwapIndexesPayload, DocumentEditionByFunction, MergeFacets, FederationOptions, SearchQueryWithIndex, Federation, FederatedSearch, FederatedSearchResult, SearchResults, SearchResultWithIndex, SimilarQuery, SimilarResult, PaginationView<serde_json::Value>, BrowseQuery, UpdateIndexRequest, IndexUid, IndexCreateRequest, KeyView, Action, CreateApiKey, UpdateStderrLogs, LogMode, GetLogs, IndexStats, Stats, HealthStatus, HealthResponse, VersionResponse, Code, ErrorType, AllTasks, TaskView, Status, DetailsView, ResponseError, Settings<Unchecked>, Settings<Checked>, TypoSettings, MinWordSizeTyposSetting, FacetingSettings, PaginationSettings, SummarizedTaskView, Kind, Network, Remote)) | ||||
|     components(schemas(PaginationView<KeyView>, PaginationView<IndexView>, IndexView, DocumentDeletionByFilter, AllBatches, BatchStats, ProgressStepView, ProgressView, BatchView, RuntimeTogglableFeatures, SwapIndexesPayload, DocumentEditionByFunction, MergeFacets, FederationOptions, SearchQueryWithIndex, Federation, FederatedSearch, FederatedSearchResult, SearchResults, SearchResultWithIndex, SimilarQuery, SimilarResult, PaginationView<serde_json::Value>, BrowseQuery, UpdateIndexRequest, IndexUid, IndexCreateRequest, KeyView, Action, CreateApiKey, UpdateStderrLogs, LogMode, GetLogs, IndexStats, Stats, HealthStatus, HealthResponse, VersionResponse, Code, ErrorType, AllTasks, TaskView, Status, DetailsView, ResponseError, Settings<Unchecked>, Settings<Checked>, TypoSettings, MinWordSizeTyposSetting, FacetingSettings, PaginationSettings, SummarizedTaskView, Kind, Network, Remote, FilterableAttributesRule, FilterableAttributesPatterns, AttributePatterns, FilterableAttributesFeatures, FilterFeatures)) | ||||
| )] | ||||
| pub struct MeilisearchApi; | ||||
|  | ||||
|   | ||||
| @@ -20,7 +20,7 @@ use meilisearch_types::milli::score_details::{ScoreDetails, ScoringStrategy}; | ||||
| use meilisearch_types::milli::vector::parsed_vectors::ExplicitVectors; | ||||
| use meilisearch_types::milli::vector::Embedder; | ||||
| use meilisearch_types::milli::{ | ||||
|     FacetValueHit, InternalError, OrderBy, SearchForFacetValues, TimeBudget, | ||||
|     FacetValueHit, InternalError, OrderBy, PatternMatch, SearchForFacetValues, TimeBudget, | ||||
| }; | ||||
| use meilisearch_types::settings::DEFAULT_PAGINATION_MAX_TOTAL_HITS; | ||||
| use meilisearch_types::{milli, Document}; | ||||
| @@ -1538,8 +1538,9 @@ pub fn perform_facet_search( | ||||
|     // If the facet string is not localized, we **ignore** the locales provided by the user because the facet data has no locale. | ||||
|     // If the user does not provide locales, we use the locales of the facet string. | ||||
|     let localized_attributes = index.localized_attributes_rules(&rtxn)?.unwrap_or_default(); | ||||
|     let localized_attributes_locales = | ||||
|         localized_attributes.into_iter().find(|attr| attr.match_str(&facet_name)); | ||||
|     let localized_attributes_locales = localized_attributes | ||||
|         .into_iter() | ||||
|         .find(|attr| attr.match_str(&facet_name) == PatternMatch::Match); | ||||
|     let locales = localized_attributes_locales.map(|attr| { | ||||
|         attr.locales | ||||
|             .into_iter() | ||||
| @@ -1885,7 +1886,7 @@ fn format_fields( | ||||
|             let locales = locales.or_else(|| { | ||||
|                 localized_attributes | ||||
|                     .iter() | ||||
|                     .find(|rule| rule.match_str(key)) | ||||
|                     .find(|rule| rule.match_str(key) == PatternMatch::Match) | ||||
|                     .map(LocalizedAttributesRule::locales) | ||||
|             }); | ||||
|  | ||||
|   | ||||
| @@ -125,6 +125,12 @@ impl Server<Owned> { | ||||
|         self.service.post("/indexes", body).await | ||||
|     } | ||||
|  | ||||
|     pub async fn delete_index(&self, uid: impl AsRef<str>) -> (Value, StatusCode) { | ||||
|         let url = format!("/indexes/{}", urlencoding::encode(uid.as_ref())); | ||||
|         let (value, code) = self.service.delete(url).await; | ||||
|         (value, code) | ||||
|     } | ||||
|  | ||||
|     pub fn index_with_encoder(&self, uid: impl AsRef<str>, encoder: Encoder) -> Index<'_> { | ||||
|         Index { | ||||
|             uid: uid.as_ref().to_string(), | ||||
|   | ||||
| @@ -636,7 +636,7 @@ async fn delete_document_by_filter() { | ||||
|         "originalFilter": "\"catto = jorts\"" | ||||
|       }, | ||||
|       "error": { | ||||
|         "message": "Index `SHARED_DOCUMENTS`: Attribute `catto` is not filterable. Available filterable attributes are: `id`, `title`.\n1:6 catto = jorts", | ||||
|         "message": "Index `SHARED_DOCUMENTS`: Attribute `catto` is not filterable. Available filterable attribute patterns are: `id`, `title`.\n1:6 catto = jorts", | ||||
|         "code": "invalid_document_filter", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_document_filter" | ||||
| @@ -738,7 +738,7 @@ async fn fetch_document_by_filter() { | ||||
|     snapshot!(code, @"400 Bad Request"); | ||||
|     snapshot!(response, @r###" | ||||
|     { | ||||
|       "message": "Attribute `doggo` is not filterable. Available filterable attributes are: `color`.\n1:6 doggo = bernese", | ||||
|       "message": "Attribute `doggo` is not filterable. Available filterable attribute patterns are: `color`.\n1:6 doggo = bernese", | ||||
|       "code": "invalid_document_filter", | ||||
|       "type": "invalid_request", | ||||
|       "link": "https://docs.meilisearch.com/errors#invalid_document_filter" | ||||
|   | ||||
| @@ -1,8 +1,10 @@ | ||||
| use meili_snap::*; | ||||
|  | ||||
| use crate::common::{shared_does_not_exists_index, Server}; | ||||
| use crate::common::{shared_does_not_exists_index, Server, DOCUMENTS, NESTED_DOCUMENTS}; | ||||
| use crate::json; | ||||
|  | ||||
| use super::test_settings_documents_indexing_swapping_and_search; | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn search_unexisting_index() { | ||||
|     let index = shared_does_not_exists_index().await; | ||||
| @@ -430,7 +432,7 @@ async fn search_non_filterable_facets() { | ||||
|     snapshot!(code, @"400 Bad Request"); | ||||
|     snapshot!(json_string!(response), @r###" | ||||
|     { | ||||
|       "message": "Invalid facet distribution, attribute `doggo` is not filterable. The available filterable attribute is `title`.", | ||||
|       "message": "Invalid facet distribution, attribute `doggo` is not filterable. The available filterable attribute pattern is `title`.", | ||||
|       "code": "invalid_search_facets", | ||||
|       "type": "invalid_request", | ||||
|       "link": "https://docs.meilisearch.com/errors#invalid_search_facets" | ||||
| @@ -441,7 +443,7 @@ async fn search_non_filterable_facets() { | ||||
|     snapshot!(code, @"400 Bad Request"); | ||||
|     snapshot!(json_string!(response), @r###" | ||||
|     { | ||||
|       "message": "Invalid facet distribution, attribute `doggo` is not filterable. The available filterable attribute is `title`.", | ||||
|       "message": "Invalid facet distribution, attribute `doggo` is not filterable. The available filterable attribute pattern is `title`.", | ||||
|       "code": "invalid_search_facets", | ||||
|       "type": "invalid_request", | ||||
|       "link": "https://docs.meilisearch.com/errors#invalid_search_facets" | ||||
| @@ -461,7 +463,7 @@ async fn search_non_filterable_facets_multiple_filterable() { | ||||
|     snapshot!(code, @"400 Bad Request"); | ||||
|     snapshot!(json_string!(response), @r###" | ||||
|     { | ||||
|       "message": "Invalid facet distribution, attribute `doggo` is not filterable. The available filterable attributes are `genres, title`.", | ||||
|       "message": "Invalid facet distribution, attribute `doggo` is not filterable. The available filterable attribute patterns are `genres, title`.", | ||||
|       "code": "invalid_search_facets", | ||||
|       "type": "invalid_request", | ||||
|       "link": "https://docs.meilisearch.com/errors#invalid_search_facets" | ||||
| @@ -472,7 +474,7 @@ async fn search_non_filterable_facets_multiple_filterable() { | ||||
|     snapshot!(code, @"400 Bad Request"); | ||||
|     snapshot!(json_string!(response), @r###" | ||||
|     { | ||||
|       "message": "Invalid facet distribution, attribute `doggo` is not filterable. The available filterable attributes are `genres, title`.", | ||||
|       "message": "Invalid facet distribution, attribute `doggo` is not filterable. The available filterable attribute patterns are `genres, title`.", | ||||
|       "code": "invalid_search_facets", | ||||
|       "type": "invalid_request", | ||||
|       "link": "https://docs.meilisearch.com/errors#invalid_search_facets" | ||||
| @@ -522,7 +524,7 @@ async fn search_non_filterable_facets_multiple_facets() { | ||||
|     snapshot!(code, @"400 Bad Request"); | ||||
|     snapshot!(json_string!(response), @r###" | ||||
|     { | ||||
|       "message": "Invalid facet distribution, attributes `doggo, neko` are not filterable. The available filterable attributes are `genres, title`.", | ||||
|       "message": "Invalid facet distribution, attributes `doggo, neko` are not filterable. The available filterable attribute patterns are `genres, title`.", | ||||
|       "code": "invalid_search_facets", | ||||
|       "type": "invalid_request", | ||||
|       "link": "https://docs.meilisearch.com/errors#invalid_search_facets" | ||||
| @@ -533,7 +535,7 @@ async fn search_non_filterable_facets_multiple_facets() { | ||||
|     snapshot!(code, @"400 Bad Request"); | ||||
|     snapshot!(json_string!(response), @r###" | ||||
|     { | ||||
|       "message": "Invalid facet distribution, attributes `doggo, neko` are not filterable. The available filterable attributes are `genres, title`.", | ||||
|       "message": "Invalid facet distribution, attributes `doggo, neko` are not filterable. The available filterable attribute patterns are `genres, title`.", | ||||
|       "code": "invalid_search_facets", | ||||
|       "type": "invalid_request", | ||||
|       "link": "https://docs.meilisearch.com/errors#invalid_search_facets" | ||||
| @@ -636,14 +638,11 @@ async fn search_bad_matching_strategy() { | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn filter_invalid_syntax_object() { | ||||
|     let server = Server::new_shared(); | ||||
|     let index = server.unique_index(); | ||||
|  | ||||
|     let (task, _code) = index.update_settings(json!({"filterableAttributes": ["title"]})).await; | ||||
|     index.wait_task(task.uid()).await.succeeded(); | ||||
|  | ||||
|     index | ||||
|         .search(json!({"filter": "title & Glass"}), |response, code| { | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"filterableAttributes": ["title"]}), | ||||
|         &json!({"filter": "title & Glass"}), | ||||
|         |response, code| { | ||||
|             snapshot!(response, @r###" | ||||
|             { | ||||
|               "message": "Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `CONTAINS`, `NOT CONTAINS`, `STARTS WITH`, `NOT STARTS WITH`, `_geoRadius`, or `_geoBoundingBox` at `title & Glass`.\n1:14 title & Glass", | ||||
| @@ -653,20 +652,18 @@ async fn filter_invalid_syntax_object() { | ||||
|             } | ||||
|             "###); | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|         }) | ||||
|         .await; | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn filter_invalid_syntax_array() { | ||||
|     let server = Server::new_shared(); | ||||
|     let index = server.unique_index(); | ||||
|  | ||||
|     let (task, _code) = index.update_settings(json!({"filterableAttributes": ["title"]})).await; | ||||
|     index.wait_task(task.uid()).await.succeeded(); | ||||
|  | ||||
|     index | ||||
|         .search(json!({"filter": ["title & Glass"]}), |response, code| { | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"filterableAttributes": ["title"]}), | ||||
|         &json!({"filter": ["title & Glass"]}), | ||||
|         |response, code| { | ||||
|             snapshot!(response, @r###" | ||||
|             { | ||||
|               "message": "Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `CONTAINS`, `NOT CONTAINS`, `STARTS WITH`, `NOT STARTS WITH`, `_geoRadius`, or `_geoBoundingBox` at `title & Glass`.\n1:14 title & Glass", | ||||
| @@ -676,206 +673,327 @@ async fn filter_invalid_syntax_array() { | ||||
|             } | ||||
|             "###); | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|         }) | ||||
|         .await; | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn filter_invalid_syntax_string() { | ||||
|     let server = Server::new_shared(); | ||||
|     let index = server.unique_index(); | ||||
|  | ||||
|     let (task, _code) = index.update_settings(json!({"filterableAttributes": ["title"]})).await; | ||||
|     index.wait_task(task.uid()).await.succeeded(); | ||||
|  | ||||
|     let expected_response = json!({ | ||||
|         "message": "Found unexpected characters at the end of the filter: `XOR title = Glass`. You probably forgot an `OR` or an `AND` rule.\n15:32 title = Glass XOR title = Glass", | ||||
|         "code": "invalid_search_filter", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|     }); | ||||
|     index | ||||
|         .search(json!({"filter": "title = Glass XOR title = Glass"}), |response, code| { | ||||
|             assert_eq!(response, expected_response); | ||||
|             assert_eq!(code, 400); | ||||
|         }) | ||||
|         .await; | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"filterableAttributes": ["title"]}), | ||||
|         &json!({"filter": "title = Glass XOR title = Glass"}), | ||||
|         |response, code| { | ||||
|             snapshot!(response, @r###" | ||||
|             { | ||||
|               "message": "Found unexpected characters at the end of the filter: `XOR title = Glass`. You probably forgot an `OR` or an `AND` rule.\n15:32 title = Glass XOR title = Glass", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn filter_invalid_attribute_array() { | ||||
|     let server = Server::new_shared(); | ||||
|     let index = server.unique_index(); | ||||
|  | ||||
|     let (task, _code) = index.update_settings(json!({"filterableAttributes": ["title"]})).await; | ||||
|     index.wait_task(task.uid()).await.succeeded(); | ||||
|  | ||||
|     let expected_response = json!({ | ||||
|         "message": format!("Index `{}`: Attribute `many` is not filterable. Available filterable attributes are: `title`.\n1:5 many = Glass", index.uid), | ||||
|         "code": "invalid_search_filter", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|     }); | ||||
|     index | ||||
|         .search(json!({"filter": ["many = Glass"]}), |response, code| { | ||||
|             assert_eq!(response, expected_response); | ||||
|             assert_eq!(code, 400); | ||||
|         }) | ||||
|         .await; | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"filterableAttributes": ["title"]}), | ||||
|         &json!({"filter": ["many = Glass"]}), | ||||
|         |response, code| { | ||||
|             snapshot!(response, @r###" | ||||
|             { | ||||
|               "message": "Index `test`: Attribute `many` is not filterable. Available filterable attribute patterns are: `title`.\n1:5 many = Glass", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn filter_invalid_attribute_string() { | ||||
|     let server = Server::new_shared(); | ||||
|     let index = server.unique_index(); | ||||
|  | ||||
|     let (task, _code) = index.update_settings(json!({"filterableAttributes": ["title"]})).await; | ||||
|     index.wait_task(task.uid()).await.succeeded(); | ||||
|  | ||||
|     let expected_response = json!({ | ||||
|         "message": format!("Index `{}`: Attribute `many` is not filterable. Available filterable attributes are: `title`.\n1:5 many = Glass", index.uid), | ||||
|         "code": "invalid_search_filter", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|     }); | ||||
|     index | ||||
|         .search(json!({"filter": "many = Glass"}), |response, code| { | ||||
|             assert_eq!(response, expected_response); | ||||
|             assert_eq!(code, 400); | ||||
|         }) | ||||
|         .await; | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"filterableAttributes": ["title"]}), | ||||
|         &json!({"filter": "many = Glass"}), | ||||
|         |response, code| { | ||||
|             snapshot!(response, @r###" | ||||
|             { | ||||
|               "message": "Index `test`: Attribute `many` is not filterable. Available filterable attribute patterns are: `title`.\n1:5 many = Glass", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn filter_reserved_geo_attribute_array() { | ||||
|     let server = Server::new_shared(); | ||||
|     let index = server.unique_index(); | ||||
|  | ||||
|     let (task, _code) = index.update_settings(json!({"filterableAttributes": ["title"]})).await; | ||||
|     index.wait_task(task.uid()).await.succeeded(); | ||||
|  | ||||
|     let expected_response = json!({ | ||||
|         "message": "`_geo` is a reserved keyword and thus can't be used as a filter expression. Use the `_geoRadius(latitude, longitude, distance)` or `_geoBoundingBox([latitude, longitude], [latitude, longitude])` built-in rules to filter on `_geo` coordinates.\n1:13 _geo = Glass", | ||||
|         "code": "invalid_search_filter", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|     }); | ||||
|     index | ||||
|         .search(json!({"filter": ["_geo = Glass"]}), |response, code| { | ||||
|             assert_eq!(response, expected_response); | ||||
|             assert_eq!(code, 400); | ||||
|         }) | ||||
|         .await; | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"filterableAttributes": ["title"]}), | ||||
|         &json!({"filter": ["_geo = Glass"]}), | ||||
|         |response, code| { | ||||
|             snapshot!(response, @r###" | ||||
|             { | ||||
|               "message": "`_geo` is a reserved keyword and thus can't be used as a filter expression. Use the `_geoRadius(latitude, longitude, distance)` or `_geoBoundingBox([latitude, longitude], [latitude, longitude])` built-in rules to filter on `_geo` coordinates.\n1:13 _geo = Glass", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn filter_reserved_geo_attribute_string() { | ||||
|     let server = Server::new_shared(); | ||||
|     let index = server.unique_index(); | ||||
|  | ||||
|     let (task, _code) = index.update_settings(json!({"filterableAttributes": ["title"]})).await; | ||||
|     index.wait_task(task.uid()).await.succeeded(); | ||||
|  | ||||
|     let expected_response = json!({ | ||||
|         "message": "`_geo` is a reserved keyword and thus can't be used as a filter expression. Use the `_geoRadius(latitude, longitude, distance)` or `_geoBoundingBox([latitude, longitude], [latitude, longitude])` built-in rules to filter on `_geo` coordinates.\n1:13 _geo = Glass", | ||||
|         "code": "invalid_search_filter", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|     }); | ||||
|     index | ||||
|         .search(json!({"filter": "_geo = Glass"}), |response, code| { | ||||
|             assert_eq!(response, expected_response); | ||||
|             assert_eq!(code, 400); | ||||
|         }) | ||||
|         .await; | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"filterableAttributes": ["title"]}), | ||||
|         &json!({"filter": "_geo = Glass"}), | ||||
|         |response, code| { | ||||
|             snapshot!(response, @r###" | ||||
|             { | ||||
|               "message": "`_geo` is a reserved keyword and thus can't be used as a filter expression. Use the `_geoRadius(latitude, longitude, distance)` or `_geoBoundingBox([latitude, longitude], [latitude, longitude])` built-in rules to filter on `_geo` coordinates.\n1:13 _geo = Glass", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn filter_reserved_attribute_array() { | ||||
|     let server = Server::new_shared(); | ||||
|     let index = server.unique_index(); | ||||
|  | ||||
|     let (task, _code) = index.update_settings(json!({"filterableAttributes": ["title"]})).await; | ||||
|     index.wait_task(task.uid()).await.succeeded(); | ||||
|  | ||||
|     let expected_response = json!({ | ||||
|         "message": "`_geoDistance` is a reserved keyword and thus can't be used as a filter expression. Use the `_geoRadius(latitude, longitude, distance)` or `_geoBoundingBox([latitude, longitude], [latitude, longitude])` built-in rules to filter on `_geo` coordinates.\n1:21 _geoDistance = Glass", | ||||
|         "code": "invalid_search_filter", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|     }); | ||||
|     index | ||||
|         .search(json!({"filter": ["_geoDistance = Glass"]}), |response, code| { | ||||
|             assert_eq!(response, expected_response); | ||||
|             assert_eq!(code, 400); | ||||
|         }) | ||||
|         .await; | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"filterableAttributes": ["title"]}), | ||||
|         &json!({"filter": ["_geoDistance = Glass"]}), | ||||
|         |response, code| { | ||||
|             snapshot!(response, @r###" | ||||
|             { | ||||
|               "message": "`_geoDistance` is a reserved keyword and thus can't be used as a filter expression. Use the `_geoRadius(latitude, longitude, distance)` or `_geoBoundingBox([latitude, longitude], [latitude, longitude])` built-in rules to filter on `_geo` coordinates.\n1:21 _geoDistance = Glass", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn filter_reserved_attribute_string() { | ||||
|     let server = Server::new_shared(); | ||||
|     let index = server.unique_index(); | ||||
|  | ||||
|     let (task, _code) = index.update_settings(json!({"filterableAttributes": ["title"]})).await; | ||||
|     index.wait_task(task.uid()).await.succeeded(); | ||||
|  | ||||
|     let expected_response = json!({ | ||||
|        "message": "`_geoDistance` is a reserved keyword and thus can't be used as a filter expression. Use the `_geoRadius(latitude, longitude, distance)` or `_geoBoundingBox([latitude, longitude], [latitude, longitude])` built-in rules to filter on `_geo` coordinates.\n1:21 _geoDistance = Glass", | ||||
|         "code": "invalid_search_filter", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|     }); | ||||
|     index | ||||
|         .search(json!({"filter": "_geoDistance = Glass"}), |response, code| { | ||||
|             assert_eq!(response, expected_response); | ||||
|             assert_eq!(code, 400); | ||||
|         }) | ||||
|         .await; | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"filterableAttributes": ["title"]}), | ||||
|         &json!({"filter": "_geoDistance = Glass"}), | ||||
|         |response, code| { | ||||
|             snapshot!(response, @r###" | ||||
|             { | ||||
|               "message": "`_geoDistance` is a reserved keyword and thus can't be used as a filter expression. Use the `_geoRadius(latitude, longitude, distance)` or `_geoBoundingBox([latitude, longitude], [latitude, longitude])` built-in rules to filter on `_geo` coordinates.\n1:21 _geoDistance = Glass", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn filter_reserved_geo_point_array() { | ||||
|     let server = Server::new_shared(); | ||||
|     let index = server.unique_index(); | ||||
|  | ||||
|     let (task, _code) = index.update_settings(json!({"filterableAttributes": ["title"]})).await; | ||||
|     index.wait_task(task.uid()).await.succeeded(); | ||||
|  | ||||
|     let expected_response = json!({ | ||||
|         "message": "`_geoPoint` is a reserved keyword and thus can't be used as a filter expression. Use the `_geoRadius(latitude, longitude, distance)` or `_geoBoundingBox([latitude, longitude], [latitude, longitude])` built-in rules to filter on `_geo` coordinates.\n1:18 _geoPoint = Glass", | ||||
|         "code": "invalid_search_filter", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|     }); | ||||
|     index | ||||
|         .search(json!({"filter": ["_geoPoint = Glass"]}), |response, code| { | ||||
|             assert_eq!(response, expected_response); | ||||
|             assert_eq!(code, 400); | ||||
|         }) | ||||
|         .await; | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"filterableAttributes": ["title"]}), | ||||
|         &json!({"filter": ["_geoPoint = Glass"]}), | ||||
|         |response, code| { | ||||
|             snapshot!(response, @r###" | ||||
|             { | ||||
|               "message": "`_geoPoint` is a reserved keyword and thus can't be used as a filter expression. Use the `_geoRadius(latitude, longitude, distance)` or `_geoBoundingBox([latitude, longitude], [latitude, longitude])` built-in rules to filter on `_geo` coordinates.\n1:18 _geoPoint = Glass", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn filter_reserved_geo_point_string() { | ||||
|     let server = Server::new_shared(); | ||||
|     let index = server.unique_index(); | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"filterableAttributes": ["title"]}), | ||||
|         &json!({"filter": "_geoPoint = Glass"}), | ||||
|         |response, code| { | ||||
|             snapshot!(response, @r###" | ||||
|             { | ||||
|               "message": "`_geoPoint` is a reserved keyword and thus can't be used as a filter expression. Use the `_geoRadius(latitude, longitude, distance)` or `_geoBoundingBox([latitude, longitude], [latitude, longitude])` built-in rules to filter on `_geo` coordinates.\n1:18 _geoPoint = Glass", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
|     let (task, _code) = index.update_settings(json!({"filterableAttributes": ["title"]})).await; | ||||
|     index.wait_task(task.uid()).await.succeeded(); | ||||
| #[actix_rt::test] | ||||
| async fn search_with_pattern_filter_settings_errors() { | ||||
|     // Check if the Equality filter works with patterns | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &NESTED_DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [{ | ||||
|             "attributePatterns": ["cattos","doggos.age"], | ||||
|             "features": { | ||||
|                 "facetSearch": false, | ||||
|                 "filter": {"equality": false, "comparison": true} | ||||
|             } | ||||
|         }]}), | ||||
|         &json!({ | ||||
|             "filter": "cattos = pésti" | ||||
|         }), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|             snapshot!(json_string!(response), @r###" | ||||
|             { | ||||
|               "message": "Index `test`: Filter operator `=` is not allowed for the attribute `cattos`.\n  - Note: allowed operators: OR, AND, NOT, <, >, <=, >=, TO, IS EMPTY, IS NULL, EXISTS.\n  - Note: field `cattos` matched rule #0 in `filterableAttributes`", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     let expected_response = json!({ | ||||
|        "message": "`_geoPoint` is a reserved keyword and thus can't be used as a filter expression. Use the `_geoRadius(latitude, longitude, distance)` or `_geoBoundingBox([latitude, longitude], [latitude, longitude])` built-in rules to filter on `_geo` coordinates.\n1:18 _geoPoint = Glass", | ||||
|         "code": "invalid_search_filter", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|     }); | ||||
|     index | ||||
|         .search(json!({"filter": "_geoPoint = Glass"}), |response, code| { | ||||
|             assert_eq!(response, expected_response); | ||||
|             assert_eq!(code, 400); | ||||
|         }) | ||||
|         .await; | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|     &NESTED_DOCUMENTS, | ||||
|     &json!({"filterableAttributes": [{ | ||||
|         "attributePatterns": ["cattos","doggos.age"], | ||||
|         "features": { | ||||
|             "facetSearch": false, | ||||
|             "filter": {"equality": false, "comparison": true} | ||||
|         } | ||||
|     }]}), | ||||
|     &json!({ | ||||
|         "filter": "cattos IN [pésti, simba]" | ||||
|     }), | ||||
|     |response, code| { | ||||
|         snapshot!(code, @"400 Bad Request"); | ||||
|         snapshot!(json_string!(response), @r###" | ||||
|         { | ||||
|           "message": "Index `test`: Filter operator `=` is not allowed for the attribute `cattos`.\n  - Note: allowed operators: OR, AND, NOT, <, >, <=, >=, TO, IS EMPTY, IS NULL, EXISTS.\n  - Note: field `cattos` matched rule #0 in `filterableAttributes`", | ||||
|           "code": "invalid_search_filter", | ||||
|           "type": "invalid_request", | ||||
|           "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|         } | ||||
|         "###); | ||||
|     }, | ||||
| ) | ||||
| .await; | ||||
|  | ||||
|     // Check if the Comparison filter works with patterns | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &NESTED_DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [{"attributePatterns": ["cattos","doggos.age"]}]}), | ||||
|         &json!({ | ||||
|             "filter": "doggos.age > 2" | ||||
|         }), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|             snapshot!(json_string!(response), @r###" | ||||
|             { | ||||
|               "message": "Index `test`: Filter operator `>` is not allowed for the attribute `doggos.age`.\n  - Note: allowed operators: OR, AND, NOT, =, !=, IN, IS EMPTY, IS NULL, EXISTS.\n  - Note: field `doggos.age` matched rule #0 in `filterableAttributes`", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &NESTED_DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [{ | ||||
|             "attributePatterns": ["cattos","doggos.age"], | ||||
|             "features": { | ||||
|                 "facetSearch": false, | ||||
|                 "filter": {"equality": true, "comparison": false} | ||||
|             } | ||||
|         }]}), | ||||
|         &json!({ | ||||
|             "filter": "doggos.age > 2" | ||||
|         }), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|             snapshot!(json_string!(response), @r###" | ||||
|             { | ||||
|               "message": "Index `test`: Filter operator `>` is not allowed for the attribute `doggos.age`.\n  - Note: allowed operators: OR, AND, NOT, =, !=, IN, IS EMPTY, IS NULL, EXISTS.\n  - Note: field `doggos.age` matched rule #0 in `filterableAttributes`", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &NESTED_DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [{ | ||||
|             "attributePatterns": ["cattos","doggos.age"], | ||||
|             "features": { | ||||
|                 "facetSearch": false, | ||||
|                 "filter": {"equality": true, "comparison": false} | ||||
|             } | ||||
|         }]}), | ||||
|         &json!({ | ||||
|             "filter": "doggos.age 2 TO 4" | ||||
|         }), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|             snapshot!(json_string!(response), @r###" | ||||
|             { | ||||
|               "message": "Index `test`: Filter operator `TO` is not allowed for the attribute `doggos.age`.\n  - Note: allowed operators: OR, AND, NOT, =, !=, IN, IS EMPTY, IS NULL, EXISTS.\n  - Note: field `doggos.age` matched rule #0 in `filterableAttributes`", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| @@ -1018,109 +1136,115 @@ async fn sort_unset_ranking_rule() { | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn search_on_unknown_field() { | ||||
|     let server = Server::new_shared(); | ||||
|     let index = server.unique_index(); | ||||
|     let (response, _code) = | ||||
|         index.update_settings_searchable_attributes(json!(["id", "title"])).await; | ||||
|     index.wait_task(response.uid()).await.succeeded(); | ||||
|  | ||||
|     let expected_response = json!({ | ||||
|         "message": format!("Index `{}`: Attribute `unknown` is not searchable. Available searchable attributes are: `id, title`.", index.uid), | ||||
|         "code": "invalid_search_attributes_to_search_on", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_search_attributes_to_search_on" | ||||
|     }); | ||||
|     index | ||||
|         .search( | ||||
|             json!({"q": "Captain Marvel", "attributesToSearchOn": ["unknown"]}), | ||||
|             |response, code| { | ||||
|                 assert_eq!(response, expected_response); | ||||
|                 assert_eq!(code, 400); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"searchableAttributes": ["id", "title"]}), | ||||
|         &json!({"q": "Captain Marvel", "attributesToSearchOn": ["unknown"]}), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|             snapshot!(response, @r###" | ||||
|             { | ||||
|               "message": "Index `test`: Attribute `unknown` is not searchable. Available searchable attributes are: `id, title`.", | ||||
|               "code": "invalid_search_attributes_to_search_on", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_attributes_to_search_on" | ||||
|             } | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn search_on_unknown_field_plus_joker() { | ||||
|     let server = Server::new_shared(); | ||||
|     let index = server.unique_index(); | ||||
|     let (response, _code) = | ||||
|         index.update_settings_searchable_attributes(json!(["id", "title"])).await; | ||||
|     index.wait_task(response.uid()).await.succeeded(); | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"searchableAttributes": ["id", "title"]}), | ||||
|         &json!({"q": "Captain Marvel", "attributesToSearchOn": ["*", "unknown"]}), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|             snapshot!(response, @r###" | ||||
|             { | ||||
|               "message": "Index `test`: Attribute `unknown` is not searchable. Available searchable attributes are: `id, title`.", | ||||
|               "code": "invalid_search_attributes_to_search_on", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_attributes_to_search_on" | ||||
|             } | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     let expected_response = json!({ | ||||
|         "message": format!("Index `{}`: Attribute `unknown` is not searchable. Available searchable attributes are: `id, title`.", index.uid), | ||||
|         "code": "invalid_search_attributes_to_search_on", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_search_attributes_to_search_on" | ||||
|     }); | ||||
|     index | ||||
|         .search( | ||||
|             json!({"q": "Captain Marvel", "attributesToSearchOn": ["*", "unknown"]}), | ||||
|             |response, code| { | ||||
|                 assert_eq!(response, expected_response); | ||||
|                 assert_eq!(code, 400); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
|  | ||||
|     index | ||||
|         .search( | ||||
|             json!({"q": "Captain Marvel", "attributesToSearchOn": ["unknown", "*"]}), | ||||
|             |response, code| { | ||||
|                 assert_eq!(response, expected_response); | ||||
|                 assert_eq!(code, 400); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"searchableAttributes": ["id", "title"]}), | ||||
|         &json!({"q": "Captain Marvel", "attributesToSearchOn": ["unknown", "*"]}), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|             snapshot!(response, @r###" | ||||
|             { | ||||
|               "message": "Index `test`: Attribute `unknown` is not searchable. Available searchable attributes are: `id, title`.", | ||||
|               "code": "invalid_search_attributes_to_search_on", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_attributes_to_search_on" | ||||
|             } | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn distinct_at_search_time() { | ||||
|     let server = Server::new_shared(); | ||||
|     let index = server.unique_index(); | ||||
|     let server = Server::new().await; | ||||
|     let index = server.index("test"); | ||||
|     let (task, _) = index.create(None).await; | ||||
|     index.wait_task(task.uid()).await.succeeded(); | ||||
|     let (response, _code) = | ||||
|         index.add_documents(json!([{"id": 1, "color": "Doggo", "machin": "Action"}]), None).await; | ||||
|     index.wait_task(response.uid()).await.succeeded(); | ||||
|  | ||||
|     let expected_response = json!({ | ||||
|         "message": format!("Index `{}`: Attribute `doggo.truc` is not filterable and thus, cannot be used as distinct attribute. This index does not have configured filterable attributes.", index.uid), | ||||
|         "code": "invalid_search_distinct", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_search_distinct" | ||||
|     }); | ||||
|     let (response, code) = | ||||
|         index.search_post(json!({"page": 0, "hitsPerPage": 2, "distinct": "doggo.truc"})).await; | ||||
|     assert_eq!(response, expected_response); | ||||
|     assert_eq!(code, 400); | ||||
|     snapshot!(code, @"400 Bad Request"); | ||||
|     snapshot!(response, @r###" | ||||
|     { | ||||
|       "message": "Index `test`: Attribute `doggo.truc` is not filterable and thus, cannot be used as distinct attribute. This index does not have configured filterable attributes.", | ||||
|       "code": "invalid_search_distinct", | ||||
|       "type": "invalid_request", | ||||
|       "link": "https://docs.meilisearch.com/errors#invalid_search_distinct" | ||||
|     } | ||||
|     "###); | ||||
|  | ||||
|     let (task, _) = index.update_settings_filterable_attributes(json!(["color", "machin"])).await; | ||||
|     index.wait_task(task.uid()).await.succeeded(); | ||||
|  | ||||
|     let expected_response = json!({ | ||||
|         "message": format!("Index `{}`: Attribute `doggo.truc` is not filterable and thus, cannot be used as distinct attribute. Available filterable attributes are: `color, machin`.", index.uid), | ||||
|         "code": "invalid_search_distinct", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_search_distinct" | ||||
|     }); | ||||
|     let (response, code) = | ||||
|         index.search_post(json!({"page": 0, "hitsPerPage": 2, "distinct": "doggo.truc"})).await; | ||||
|     assert_eq!(response, expected_response); | ||||
|     assert_eq!(code, 400); | ||||
|     snapshot!(code, @"400 Bad Request"); | ||||
|     snapshot!(response, @r###" | ||||
|     { | ||||
|       "message": "Index `test`: Attribute `doggo.truc` is not filterable and thus, cannot be used as distinct attribute. Available filterable attributes patterns are: `color, machin`.", | ||||
|       "code": "invalid_search_distinct", | ||||
|       "type": "invalid_request", | ||||
|       "link": "https://docs.meilisearch.com/errors#invalid_search_distinct" | ||||
|     } | ||||
|     "###); | ||||
|  | ||||
|     let (task, _) = index.update_settings_displayed_attributes(json!(["color"])).await; | ||||
|     index.wait_task(task.uid()).await.succeeded(); | ||||
|  | ||||
|     let expected_response = json!({ | ||||
|         "message": format!("Index `{}`: Attribute `doggo.truc` is not filterable and thus, cannot be used as distinct attribute. Available filterable attributes are: `color, <..hidden-attributes>`.", index.uid), | ||||
|         "code": "invalid_search_distinct", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_search_distinct" | ||||
|     }); | ||||
|     let (response, code) = | ||||
|         index.search_post(json!({"page": 0, "hitsPerPage": 2, "distinct": "doggo.truc"})).await; | ||||
|     assert_eq!(response, expected_response); | ||||
|     assert_eq!(code, 400); | ||||
|     snapshot!(code, @"400 Bad Request"); | ||||
|     snapshot!(response, @r###" | ||||
|     { | ||||
|       "message": "Index `test`: Attribute `doggo.truc` is not filterable and thus, cannot be used as distinct attribute. Available filterable attributes patterns are: `color, <..hidden-attributes>`.", | ||||
|       "code": "invalid_search_distinct", | ||||
|       "type": "invalid_request", | ||||
|       "link": "https://docs.meilisearch.com/errors#invalid_search_distinct" | ||||
|     } | ||||
|     "###); | ||||
|  | ||||
|     let (response, code) = | ||||
|         index.search_post(json!({"page": 0, "hitsPerPage": 2, "distinct": true})).await; | ||||
|   | ||||
| @@ -1,7 +1,9 @@ | ||||
| use meili_snap::snapshot; | ||||
| use meilisearch::Opt; | ||||
| use once_cell::sync::Lazy; | ||||
| use tempfile::TempDir; | ||||
|  | ||||
| use crate::common::{Server, Value}; | ||||
| use crate::common::{default_settings, Server, Value, NESTED_DOCUMENTS}; | ||||
| use crate::json; | ||||
|  | ||||
| static DOCUMENTS: Lazy<Value> = Lazy::new(|| { | ||||
| @@ -34,6 +36,62 @@ static DOCUMENTS: Lazy<Value> = Lazy::new(|| { | ||||
|     ]) | ||||
| }); | ||||
|  | ||||
| async fn test_settings_documents_indexing_swapping_and_facet_search( | ||||
|     documents: &Value, | ||||
|     settings: &Value, | ||||
|     query: &Value, | ||||
|     test: impl Fn(Value, actix_http::StatusCode) + std::panic::UnwindSafe + Clone, | ||||
| ) { | ||||
|     let temp = TempDir::new().unwrap(); | ||||
|     let server = Server::new_with_options(Opt { ..default_settings(temp.path()) }).await.unwrap(); | ||||
|  | ||||
|     eprintln!("Documents -> Settings -> test"); | ||||
|     let index = server.index("test"); | ||||
|  | ||||
|     let (task, code) = index.add_documents(documents.clone(), None).await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = index.wait_task(task.uid()).await; | ||||
|     assert!(response.is_success(), "{:?}", response); | ||||
|  | ||||
|     let (task, code) = index.update_settings(settings.clone()).await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = index.wait_task(task.uid()).await; | ||||
|     assert!(response.is_success(), "{:?}", response); | ||||
|  | ||||
|     let (response, code) = index.facet_search(query.clone()).await; | ||||
|     insta::allow_duplicates! { | ||||
|         test(response, code); | ||||
|     } | ||||
|  | ||||
|     let (task, code) = server.delete_index("test").await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = server.wait_task(task.uid()).await; | ||||
|     assert!(response.is_success(), "{:?}", response); | ||||
|  | ||||
|     eprintln!("Settings -> Documents -> test"); | ||||
|     let index = server.index("test"); | ||||
|  | ||||
|     let (task, code) = index.update_settings(settings.clone()).await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = index.wait_task(task.uid()).await; | ||||
|     assert!(response.is_success(), "{:?}", response); | ||||
|  | ||||
|     let (task, code) = index.add_documents(documents.clone(), None).await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = index.wait_task(task.uid()).await; | ||||
|     assert!(response.is_success(), "{:?}", response); | ||||
|  | ||||
|     let (response, code) = index.facet_search(query.clone()).await; | ||||
|     insta::allow_duplicates! { | ||||
|         test(response, code); | ||||
|     } | ||||
|  | ||||
|     let (task, code) = server.delete_index("test").await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = server.wait_task(task.uid()).await; | ||||
|     assert!(response.is_success(), "{:?}", response); | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn simple_facet_search() { | ||||
|     let server = Server::new().await; | ||||
| @@ -436,3 +494,124 @@ async fn deactivate_facet_search_add_documents_and_reset_facet_search() { | ||||
|     assert_eq!(code, 200, "{}", response); | ||||
|     assert_eq!(dbg!(response)["facetHits"].as_array().unwrap().len(), 2); | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn facet_search_with_filterable_attributes_rules() { | ||||
|     test_settings_documents_indexing_swapping_and_facet_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"filterableAttributes": ["genres"]}), | ||||
|         &json!({"facetName": "genres", "facetQuery": "a"}), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"200 OK"); | ||||
|             snapshot!(response["facetHits"], @r###"[{"value":"Action","count":3},{"value":"Adventure","count":2}]"###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_facet_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [{"attributePatterns": ["genres"], "features": {"facetSearch": true, "filter": {"equality": false, "comparison": false}}}]}), | ||||
|         &json!({"facetName": "genres", "facetQuery": "a"}), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"200 OK"); | ||||
|             snapshot!(response["facetHits"], @r###"[{"value":"Action","count":3},{"value":"Adventure","count":2}]"###); | ||||
|         }, | ||||
|     ).await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_facet_search( | ||||
|         &NESTED_DOCUMENTS, | ||||
|         &json!({"filterableAttributes": ["doggos.name"]}), | ||||
|         &json!({"facetName": "doggos.name", "facetQuery": "b"}), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"200 OK"); | ||||
|             snapshot!(response["facetHits"], @r###"[{"value":"bobby","count":1},{"value":"buddy","count":1}]"###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_facet_search( | ||||
|         &NESTED_DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [{"attributePatterns": ["doggos.name"], "features": {"facetSearch": true, "filter": {"equality": false, "comparison": false}}}]}), | ||||
|         &json!({"facetName": "doggos.name", "facetQuery": "b"}), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"200 OK"); | ||||
|             snapshot!(response["facetHits"], @r###"[{"value":"bobby","count":1},{"value":"buddy","count":1}]"###); | ||||
|         }, | ||||
|     ).await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn facet_search_with_filterable_attributes_rules_errors() { | ||||
|     test_settings_documents_indexing_swapping_and_facet_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"filterableAttributes": ["genres"]}), | ||||
|         &json!({"facetName": "invalid", "facetQuery": "a"}), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|             snapshot!(response["message"], @r###""Attribute `invalid` is not facet-searchable. Available facet-searchable attributes patterns are: `genres`. To make it facet-searchable add it to the `filterableAttributes` index settings.""###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_facet_search( | ||||
|       &DOCUMENTS, | ||||
|       &json!({"filterableAttributes": [{"attributePatterns": ["genres"]}]}), | ||||
|       &json!({"facetName": "genres", "facetQuery": "a"}), | ||||
|       |response, code| { | ||||
|           snapshot!(code, @"400 Bad Request"); | ||||
|           snapshot!(response["message"], @r###""Attribute `genres` is not facet-searchable. This index does not have configured facet-searchable attributes. To make it facet-searchable add it to the `filterableAttributes` index settings.""###); | ||||
|       }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_facet_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [{"attributePatterns": ["genres"], "features": {"facetSearch": false, "filter": {"equality": true, "comparison": true}}}]}), | ||||
|         &json!({"facetName": "genres", "facetQuery": "a"}), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|             snapshot!(response["message"], @r###""Attribute `genres` is not facet-searchable. This index does not have configured facet-searchable attributes. To make it facet-searchable add it to the `filterableAttributes` index settings.""###); | ||||
|         }, | ||||
|     ).await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_facet_search( | ||||
|         &DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [{"attributePatterns": ["genres"], "features": {"facetSearch": false, "filter": {"equality": false, "comparison": false}}}]}), | ||||
|         &json!({"facetName": "genres", "facetQuery": "a"}), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|             snapshot!(response["message"], @r###""Attribute `genres` is not facet-searchable. This index does not have configured facet-searchable attributes. To make it facet-searchable add it to the `filterableAttributes` index settings.""###); | ||||
|         }, | ||||
|     ).await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_facet_search( | ||||
|         &NESTED_DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [{"attributePatterns": ["doggos.name"]}]}), | ||||
|         &json!({"facetName": "invalid.name", "facetQuery": "b"}), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|             snapshot!(response["message"], @r###""Attribute `invalid.name` is not facet-searchable. This index does not have configured facet-searchable attributes. To make it facet-searchable add it to the `filterableAttributes` index settings.""###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_facet_search( | ||||
|         &NESTED_DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [{"attributePatterns": ["doggos.name"], "features": {"facetSearch": false, "filter": {"equality": true, "comparison": true}}}]}), | ||||
|         &json!({"facetName": "doggos.name", "facetQuery": "b"}), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|             snapshot!(response["message"], @r###""Attribute `doggos.name` is not facet-searchable. This index does not have configured facet-searchable attributes. To make it facet-searchable add it to the `filterableAttributes` index settings.""###); | ||||
|         }, | ||||
|     ).await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_facet_search( | ||||
|         &NESTED_DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [{"attributePatterns": ["doggos.name"], "features": {"facetSearch": false, "filter": {"equality": false, "comparison": false}}}]}), | ||||
|         &json!({"facetName": "doggos.name", "facetQuery": "b"}), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|             snapshot!(response["message"], @r###""Attribute `doggos.name` is not facet-searchable. This index does not have configured facet-searchable attributes. To make it facet-searchable add it to the `filterableAttributes` index settings.""###); | ||||
|         }, | ||||
|     ).await; | ||||
| } | ||||
|   | ||||
							
								
								
									
										758
									
								
								crates/meilisearch/tests/search/filters.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										758
									
								
								crates/meilisearch/tests/search/filters.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,758 @@ | ||||
| use meili_snap::{json_string, snapshot}; | ||||
| use meilisearch::Opt; | ||||
| use tempfile::TempDir; | ||||
|  | ||||
| use super::test_settings_documents_indexing_swapping_and_search; | ||||
| use crate::{ | ||||
|     common::{default_settings, shared_index_with_documents, Server, DOCUMENTS, NESTED_DOCUMENTS}, | ||||
|     json, | ||||
| }; | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn search_with_filter_string_notation() { | ||||
|     let server = Server::new().await; | ||||
|     let index = server.index("test"); | ||||
|  | ||||
|     let (_, code) = index.update_settings(json!({"filterableAttributes": ["title"]})).await; | ||||
|     meili_snap::snapshot!(code, @"202 Accepted"); | ||||
|  | ||||
|     let documents = DOCUMENTS.clone(); | ||||
|     let (task, code) = index.add_documents(documents, None).await; | ||||
|     meili_snap::snapshot!(code, @"202 Accepted"); | ||||
|     let res = index.wait_task(task.uid()).await; | ||||
|     meili_snap::snapshot!(res["status"], @r###""succeeded""###); | ||||
|  | ||||
|     index | ||||
|         .search( | ||||
|             json!({ | ||||
|                 "filter": "title = Gläss" | ||||
|             }), | ||||
|             |response, code| { | ||||
|                 assert_eq!(code, 200, "{}", response); | ||||
|                 assert_eq!(response["hits"].as_array().unwrap().len(), 1); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
|  | ||||
|     let index = server.index("nested"); | ||||
|  | ||||
|     let (_, code) = | ||||
|         index.update_settings(json!({"filterableAttributes": ["cattos", "doggos.age"]})).await; | ||||
|     meili_snap::snapshot!(code, @"202 Accepted"); | ||||
|  | ||||
|     let documents = NESTED_DOCUMENTS.clone(); | ||||
|     let (task, code) = index.add_documents(documents, None).await; | ||||
|     meili_snap::snapshot!(code, @"202 Accepted"); | ||||
|     let res = index.wait_task(task.uid()).await; | ||||
|     meili_snap::snapshot!(res["status"], @r###""succeeded""###); | ||||
|  | ||||
|     index | ||||
|         .search( | ||||
|             json!({ | ||||
|                 "filter": "cattos = pésti" | ||||
|             }), | ||||
|             |response, code| { | ||||
|                 assert_eq!(code, 200, "{}", response); | ||||
|                 assert_eq!(response["hits"].as_array().unwrap().len(), 1); | ||||
|                 assert_eq!(response["hits"][0]["id"], json!(852)); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
|  | ||||
|     index | ||||
|         .search( | ||||
|             json!({ | ||||
|                 "filter": "doggos.age > 5" | ||||
|             }), | ||||
|             |response, code| { | ||||
|                 assert_eq!(code, 200, "{}", response); | ||||
|                 assert_eq!(response["hits"].as_array().unwrap().len(), 2); | ||||
|                 assert_eq!(response["hits"][0]["id"], json!(654)); | ||||
|                 assert_eq!(response["hits"][1]["id"], json!(951)); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn search_with_filter_array_notation() { | ||||
|     let index = shared_index_with_documents().await; | ||||
|     let (response, code) = index | ||||
|         .search_post(json!({ | ||||
|             "filter": ["title = Gläss"] | ||||
|         })) | ||||
|         .await; | ||||
|     assert_eq!(code, 200, "{}", response); | ||||
|     assert_eq!(response["hits"].as_array().unwrap().len(), 1); | ||||
|  | ||||
|     let (response, code) = index | ||||
|         .search_post(json!({ | ||||
|             "filter": [["title = Gläss", "title = \"Shazam!\"", "title = \"Escape Room\""]] | ||||
|         })) | ||||
|         .await; | ||||
|     assert_eq!(code, 200, "{}", response); | ||||
|     assert_eq!(response["hits"].as_array().unwrap().len(), 3); | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn search_with_contains_filter() { | ||||
|     let temp = TempDir::new().unwrap(); | ||||
|     let server = Server::new_with_options(Opt { | ||||
|         experimental_contains_filter: true, | ||||
|         ..default_settings(temp.path()) | ||||
|     }) | ||||
|     .await | ||||
|     .unwrap(); | ||||
|     let index = server.index("movies"); | ||||
|  | ||||
|     index.update_settings(json!({"filterableAttributes": ["title"]})).await; | ||||
|  | ||||
|     let documents = DOCUMENTS.clone(); | ||||
|     let (request, _code) = index.add_documents(documents, None).await; | ||||
|     index.wait_task(request.uid()).await.succeeded(); | ||||
|  | ||||
|     let (response, code) = index | ||||
|         .search_post(json!({ | ||||
|             "filter": "title CONTAINS cap" | ||||
|         })) | ||||
|         .await; | ||||
|     assert_eq!(code, 200, "{}", response); | ||||
|     assert_eq!(response["hits"].as_array().unwrap().len(), 2); | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn search_with_pattern_filter_settings() { | ||||
|     // Check if the Equality filter works with patterns | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &NESTED_DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [{"attributePatterns": ["cattos","doggos.age"]}]}), | ||||
|         &json!({ | ||||
|             "filter": "cattos = pésti" | ||||
|         }), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"200 OK"); | ||||
|             snapshot!(json_string!(response["hits"]), @r###" | ||||
|             [ | ||||
|               { | ||||
|                 "id": 852, | ||||
|                 "father": "jean", | ||||
|                 "mother": "michelle", | ||||
|                 "doggos": [ | ||||
|                   { | ||||
|                     "name": "bobby", | ||||
|                     "age": 2 | ||||
|                   }, | ||||
|                   { | ||||
|                     "name": "buddy", | ||||
|                     "age": 4 | ||||
|                   } | ||||
|                 ], | ||||
|                 "cattos": "pésti" | ||||
|               } | ||||
|             ] | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &NESTED_DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [{ | ||||
|             "attributePatterns": ["cattos","doggos.age"], | ||||
|             "features": { | ||||
|                 "facetSearch": false, | ||||
|                 "filter": {"equality": true, "comparison": false} | ||||
|             } | ||||
|         }]}), | ||||
|         &json!({ | ||||
|             "filter": "cattos = pésti" | ||||
|         }), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"200 OK"); | ||||
|             snapshot!(json_string!(response["hits"]), @r###" | ||||
|             [ | ||||
|               { | ||||
|                 "id": 852, | ||||
|                 "father": "jean", | ||||
|                 "mother": "michelle", | ||||
|                 "doggos": [ | ||||
|                   { | ||||
|                     "name": "bobby", | ||||
|                     "age": 2 | ||||
|                   }, | ||||
|                   { | ||||
|                     "name": "buddy", | ||||
|                     "age": 4 | ||||
|                   } | ||||
|                 ], | ||||
|                 "cattos": "pésti" | ||||
|               } | ||||
|             ] | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     // Check if the Comparison filter works with patterns | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &NESTED_DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [{ | ||||
|             "attributePatterns": ["cattos","doggos.age"], | ||||
|             "features": { | ||||
|                 "facetSearch": false, | ||||
|                 "filter": {"equality": false, "comparison": true} | ||||
|             } | ||||
|         }]}), | ||||
|         &json!({ | ||||
|             "filter": "doggos.age > 2" | ||||
|         }), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"200 OK"); | ||||
|             snapshot!(json_string!(response["hits"]), @r###" | ||||
|             [ | ||||
|               { | ||||
|                 "id": 852, | ||||
|                 "father": "jean", | ||||
|                 "mother": "michelle", | ||||
|                 "doggos": [ | ||||
|                   { | ||||
|                     "name": "bobby", | ||||
|                     "age": 2 | ||||
|                   }, | ||||
|                   { | ||||
|                     "name": "buddy", | ||||
|                     "age": 4 | ||||
|                   } | ||||
|                 ], | ||||
|                 "cattos": "pésti" | ||||
|               }, | ||||
|               { | ||||
|                 "id": 654, | ||||
|                 "father": "pierre", | ||||
|                 "mother": "sabine", | ||||
|                 "doggos": [ | ||||
|                   { | ||||
|                     "name": "gros bill", | ||||
|                     "age": 8 | ||||
|                   } | ||||
|                 ], | ||||
|                 "cattos": [ | ||||
|                   "simba", | ||||
|                   "pestiféré" | ||||
|                 ] | ||||
|               }, | ||||
|               { | ||||
|                 "id": 951, | ||||
|                 "father": "jean-baptiste", | ||||
|                 "mother": "sophie", | ||||
|                 "doggos": [ | ||||
|                   { | ||||
|                     "name": "turbo", | ||||
|                     "age": 5 | ||||
|                   }, | ||||
|                   { | ||||
|                     "name": "fast", | ||||
|                     "age": 6 | ||||
|                   } | ||||
|                 ], | ||||
|                 "cattos": [ | ||||
|                   "moumoute", | ||||
|                   "gomez" | ||||
|                 ] | ||||
|               } | ||||
|             ] | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn search_with_pattern_filter_settings_scenario_1() { | ||||
|     let temp = TempDir::new().unwrap(); | ||||
|     let server = Server::new_with_options(Opt { ..default_settings(temp.path()) }).await.unwrap(); | ||||
|  | ||||
|     eprintln!("Documents -> Settings -> test"); | ||||
|     let index = server.index("test"); | ||||
|  | ||||
|     let (task, code) = index.add_documents(NESTED_DOCUMENTS.clone(), None).await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = index.wait_task(task.uid()).await; | ||||
|     snapshot!(response["status"], @r###""succeeded""###); | ||||
|  | ||||
|     let (task, code) = index | ||||
|         .update_settings(json!({"filterableAttributes": [{ | ||||
|             "attributePatterns": ["cattos","doggos.age"], | ||||
|             "features": { | ||||
|                 "facetSearch": false, | ||||
|                 "filter": {"equality": true, "comparison": false} | ||||
|             } | ||||
|         }]})) | ||||
|         .await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = index.wait_task(task.uid()).await; | ||||
|     snapshot!(response["status"], @r###""succeeded""###); | ||||
|  | ||||
|     // Check if the Equality filter works | ||||
|     index | ||||
|         .search( | ||||
|             json!({ | ||||
|                 "filter": "cattos = pésti" | ||||
|             }), | ||||
|             |response, code| { | ||||
|                 snapshot!(code, @"200 OK"); | ||||
|                 snapshot!(json_string!(response["hits"]), @r###" | ||||
|                 [ | ||||
|                   { | ||||
|                     "id": 852, | ||||
|                     "father": "jean", | ||||
|                     "mother": "michelle", | ||||
|                     "doggos": [ | ||||
|                       { | ||||
|                         "name": "bobby", | ||||
|                         "age": 2 | ||||
|                       }, | ||||
|                       { | ||||
|                         "name": "buddy", | ||||
|                         "age": 4 | ||||
|                       } | ||||
|                     ], | ||||
|                     "cattos": "pésti" | ||||
|                   } | ||||
|                 ] | ||||
|                 "###); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
|  | ||||
|     // Check if the Comparison filter returns an error | ||||
|     index | ||||
|         .search( | ||||
|             json!({ | ||||
|                 "filter": "doggos.age > 2" | ||||
|             }), | ||||
|             |response, code| { | ||||
|                 snapshot!(code, @"400 Bad Request"); | ||||
|                 snapshot!(json_string!(response), @r###" | ||||
|                 { | ||||
|                   "message": "Index `test`: Filter operator `>` is not allowed for the attribute `doggos.age`.\n  - Note: allowed operators: OR, AND, NOT, =, !=, IN, IS EMPTY, IS NULL, EXISTS.\n  - Note: field `doggos.age` matched rule #0 in `filterableAttributes`", | ||||
|                   "code": "invalid_search_filter", | ||||
|                   "type": "invalid_request", | ||||
|                   "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|                 } | ||||
|                 "###); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
|  | ||||
|     // Update the settings activate comparison filter | ||||
|     let (task, code) = index | ||||
|         .update_settings(json!({"filterableAttributes": [{ | ||||
|             "attributePatterns": ["cattos","doggos.age"], | ||||
|             "features": { | ||||
|                 "facetSearch": false, | ||||
|                 "filter": {"equality": true, "comparison": true} | ||||
|             } | ||||
|         }]})) | ||||
|         .await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = index.wait_task(task.uid()).await; | ||||
|     snapshot!(response["status"], @r###""succeeded""###); | ||||
|  | ||||
|     // Check if the Equality filter works | ||||
|     index | ||||
|         .search( | ||||
|             json!({ | ||||
|                 "filter": "cattos = pésti" | ||||
|             }), | ||||
|             |response, code| { | ||||
|                 snapshot!(code, @"200 OK"); | ||||
|                 snapshot!(json_string!(response["hits"]), @r###" | ||||
|                 [ | ||||
|                   { | ||||
|                     "id": 852, | ||||
|                     "father": "jean", | ||||
|                     "mother": "michelle", | ||||
|                     "doggos": [ | ||||
|                       { | ||||
|                         "name": "bobby", | ||||
|                         "age": 2 | ||||
|                       }, | ||||
|                       { | ||||
|                         "name": "buddy", | ||||
|                         "age": 4 | ||||
|                       } | ||||
|                     ], | ||||
|                     "cattos": "pésti" | ||||
|                   } | ||||
|                 ] | ||||
|                 "###); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
|  | ||||
|     // Check if the Comparison filter works | ||||
|     index | ||||
|         .search( | ||||
|             json!({ | ||||
|                 "filter": "doggos.age > 2" | ||||
|             }), | ||||
|             |response, code| { | ||||
|                 snapshot!(code, @"200 OK"); | ||||
|                 snapshot!(json_string!(response["hits"]), @r###" | ||||
|                 [ | ||||
|                   { | ||||
|                     "id": 852, | ||||
|                     "father": "jean", | ||||
|                     "mother": "michelle", | ||||
|                     "doggos": [ | ||||
|                       { | ||||
|                         "name": "bobby", | ||||
|                         "age": 2 | ||||
|                       }, | ||||
|                       { | ||||
|                         "name": "buddy", | ||||
|                         "age": 4 | ||||
|                       } | ||||
|                     ], | ||||
|                     "cattos": "pésti" | ||||
|                   }, | ||||
|                   { | ||||
|                     "id": 654, | ||||
|                     "father": "pierre", | ||||
|                     "mother": "sabine", | ||||
|                     "doggos": [ | ||||
|                       { | ||||
|                         "name": "gros bill", | ||||
|                         "age": 8 | ||||
|                       } | ||||
|                     ], | ||||
|                     "cattos": [ | ||||
|                       "simba", | ||||
|                       "pestiféré" | ||||
|                     ] | ||||
|                   }, | ||||
|                   { | ||||
|                     "id": 951, | ||||
|                     "father": "jean-baptiste", | ||||
|                     "mother": "sophie", | ||||
|                     "doggos": [ | ||||
|                       { | ||||
|                         "name": "turbo", | ||||
|                         "age": 5 | ||||
|                       }, | ||||
|                       { | ||||
|                         "name": "fast", | ||||
|                         "age": 6 | ||||
|                       } | ||||
|                     ], | ||||
|                     "cattos": [ | ||||
|                       "moumoute", | ||||
|                       "gomez" | ||||
|                     ] | ||||
|                   } | ||||
|                 ] | ||||
|                 "###); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
|  | ||||
|     // Update the settings deactivate equality filter | ||||
|     let (task, code) = index | ||||
|         .update_settings(json!({"filterableAttributes": [{ | ||||
|             "attributePatterns": ["cattos","doggos.age"], | ||||
|             "features": { | ||||
|                 "facetSearch": false, | ||||
|                 "filter": {"equality": false, "comparison": true} | ||||
|             } | ||||
|         }]})) | ||||
|         .await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = index.wait_task(task.uid()).await; | ||||
|     snapshot!(response["status"], @r###""succeeded""###); | ||||
|  | ||||
|     // Check if the Equality filter returns an error | ||||
|     index | ||||
|         .search( | ||||
|             json!({ | ||||
|                 "filter": "cattos = pésti" | ||||
|             }), | ||||
|             |response, code| { | ||||
|                 snapshot!(code, @"400 Bad Request"); | ||||
|                 snapshot!(json_string!(response), @r###" | ||||
|                 { | ||||
|                   "message": "Index `test`: Filter operator `=` is not allowed for the attribute `cattos`.\n  - Note: allowed operators: OR, AND, NOT, <, >, <=, >=, TO, IS EMPTY, IS NULL, EXISTS.\n  - Note: field `cattos` matched rule #0 in `filterableAttributes`", | ||||
|                   "code": "invalid_search_filter", | ||||
|                   "type": "invalid_request", | ||||
|                   "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|                 } | ||||
|                 "###); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
|  | ||||
|     // Check if the Comparison filter works | ||||
|     index | ||||
|         .search( | ||||
|             json!({ | ||||
|                 "filter": "doggos.age > 2" | ||||
|             }), | ||||
|             |response, code| { | ||||
|                 snapshot!(code, @"200 OK"); | ||||
|                 snapshot!(json_string!(response["hits"]), @r###" | ||||
|                 [ | ||||
|                   { | ||||
|                     "id": 852, | ||||
|                     "father": "jean", | ||||
|                     "mother": "michelle", | ||||
|                     "doggos": [ | ||||
|                       { | ||||
|                         "name": "bobby", | ||||
|                         "age": 2 | ||||
|                       }, | ||||
|                       { | ||||
|                         "name": "buddy", | ||||
|                         "age": 4 | ||||
|                       } | ||||
|                     ], | ||||
|                     "cattos": "pésti" | ||||
|                   }, | ||||
|                   { | ||||
|                     "id": 654, | ||||
|                     "father": "pierre", | ||||
|                     "mother": "sabine", | ||||
|                     "doggos": [ | ||||
|                       { | ||||
|                         "name": "gros bill", | ||||
|                         "age": 8 | ||||
|                       } | ||||
|                     ], | ||||
|                     "cattos": [ | ||||
|                       "simba", | ||||
|                       "pestiféré" | ||||
|                     ] | ||||
|                   }, | ||||
|                   { | ||||
|                     "id": 951, | ||||
|                     "father": "jean-baptiste", | ||||
|                     "mother": "sophie", | ||||
|                     "doggos": [ | ||||
|                       { | ||||
|                         "name": "turbo", | ||||
|                         "age": 5 | ||||
|                       }, | ||||
|                       { | ||||
|                         "name": "fast", | ||||
|                         "age": 6 | ||||
|                       } | ||||
|                     ], | ||||
|                     "cattos": [ | ||||
|                       "moumoute", | ||||
|                       "gomez" | ||||
|                     ] | ||||
|                   } | ||||
|                 ] | ||||
|                 "###); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
|  | ||||
|     // rollback the settings | ||||
|     let (task, code) = index | ||||
|         .update_settings(json!({"filterableAttributes": [{ | ||||
|             "attributePatterns": ["cattos","doggos.age"], | ||||
|             "features": { | ||||
|                 "facetSearch": false, | ||||
|                 "filter": {"equality": true, "comparison": false} | ||||
|             } | ||||
|         }]})) | ||||
|         .await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = index.wait_task(task.uid()).await; | ||||
|     snapshot!(response["status"], @r###""succeeded""###); | ||||
|  | ||||
|     // Check if the Equality filter works | ||||
|     index | ||||
|         .search( | ||||
|             json!({ | ||||
|                 "filter": "cattos = pésti" | ||||
|             }), | ||||
|             |response, code| { | ||||
|                 snapshot!(code, @"200 OK"); | ||||
|                 snapshot!(json_string!(response["hits"]), @r###" | ||||
|                 [ | ||||
|                   { | ||||
|                     "id": 852, | ||||
|                     "father": "jean", | ||||
|                     "mother": "michelle", | ||||
|                     "doggos": [ | ||||
|                       { | ||||
|                         "name": "bobby", | ||||
|                         "age": 2 | ||||
|                       }, | ||||
|                       { | ||||
|                         "name": "buddy", | ||||
|                         "age": 4 | ||||
|                       } | ||||
|                     ], | ||||
|                     "cattos": "pésti" | ||||
|                   } | ||||
|                 ] | ||||
|                 "###); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
|  | ||||
|     // Check if the Comparison filter returns an error | ||||
|     index | ||||
|         .search( | ||||
|             json!({ | ||||
|                 "filter": "doggos.age > 2" | ||||
|             }), | ||||
|             |response, code| { | ||||
|                 snapshot!(code, @"400 Bad Request"); | ||||
|                 snapshot!(json_string!(response), @r###" | ||||
|                 { | ||||
|                   "message": "Index `test`: Filter operator `>` is not allowed for the attribute `doggos.age`.\n  - Note: allowed operators: OR, AND, NOT, =, !=, IN, IS EMPTY, IS NULL, EXISTS.\n  - Note: field `doggos.age` matched rule #0 in `filterableAttributes`", | ||||
|                   "code": "invalid_search_filter", | ||||
|                   "type": "invalid_request", | ||||
|                   "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|                 } | ||||
|                 "###); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn test_filterable_attributes_priority() { | ||||
|     // Test that the filterable attributes priority is respected | ||||
|  | ||||
|     // check if doggos.name is filterable | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &NESTED_DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [ | ||||
|             // deactivated filter | ||||
|             {"attributePatterns": ["doggos.a*"], "features": {"facetSearch": false, "filter": {"equality": false, "comparison": false}}}, | ||||
|             // activated filter | ||||
|             {"attributePatterns": ["doggos.*"]}, | ||||
|         ]}), | ||||
|         &json!({ | ||||
|             "filter": "doggos.name = bobby" | ||||
|         }), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"200 OK"); | ||||
|             snapshot!(json_string!(response["hits"]), @r###" | ||||
|             [ | ||||
|               { | ||||
|                 "id": 852, | ||||
|                 "father": "jean", | ||||
|                 "mother": "michelle", | ||||
|                 "doggos": [ | ||||
|                   { | ||||
|                     "name": "bobby", | ||||
|                     "age": 2 | ||||
|                   }, | ||||
|                   { | ||||
|                     "name": "buddy", | ||||
|                     "age": 4 | ||||
|                   } | ||||
|                 ], | ||||
|                 "cattos": "pésti" | ||||
|               } | ||||
|             ] | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     // check if doggos.name is filterable 2 | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &NESTED_DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [ | ||||
|             // deactivated filter | ||||
|             {"attributePatterns": ["doggos"], "features": {"facetSearch": false, "filter": {"equality": false, "comparison": false}}}, | ||||
|             // activated filter | ||||
|             {"attributePatterns": ["doggos.*"]}, | ||||
|         ]}), | ||||
|         &json!({ | ||||
|             "filter": "doggos.name = bobby" | ||||
|         }), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"200 OK"); | ||||
|             snapshot!(json_string!(response["hits"]), @r###" | ||||
|             [ | ||||
|               { | ||||
|                 "id": 852, | ||||
|                 "father": "jean", | ||||
|                 "mother": "michelle", | ||||
|                 "doggos": [ | ||||
|                   { | ||||
|                     "name": "bobby", | ||||
|                     "age": 2 | ||||
|                   }, | ||||
|                   { | ||||
|                     "name": "buddy", | ||||
|                     "age": 4 | ||||
|                   } | ||||
|                 ], | ||||
|                 "cattos": "pésti" | ||||
|               } | ||||
|             ] | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     // check if doggos.age is not filterable | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &NESTED_DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [ | ||||
|             // deactivated filter | ||||
|             {"attributePatterns": ["doggos.a*"], "features": {"facetSearch": false, "filter": {"equality": false, "comparison": false}}}, | ||||
|             // activated filter | ||||
|             {"attributePatterns": ["doggos.*"]}, | ||||
|         ]}), | ||||
|         &json!({ | ||||
|             "filter": "doggos.age > 2" | ||||
|         }), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|             snapshot!(json_string!(response), @r###" | ||||
|             { | ||||
|               "message": "Index `test`: Attribute `doggos.age` is not filterable. Available filterable attribute patterns are: `doggos.*`.\n1:11 doggos.age > 2", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     // check if doggos is not filterable | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &NESTED_DOCUMENTS, | ||||
|         &json!({"filterableAttributes": [ | ||||
|             // deactivated filter | ||||
|             {"attributePatterns": ["doggos"], "features": {"facetSearch": false, "filter": {"equality": false, "comparison": false}}}, | ||||
|             // activated filter | ||||
|             {"attributePatterns": ["doggos.*"]}, | ||||
|         ]}), | ||||
|         &json!({ | ||||
|             "filter": "doggos EXISTS" | ||||
|         }), | ||||
|         |response, code| { | ||||
|             snapshot!(code, @"400 Bad Request"); | ||||
|             snapshot!(json_string!(response), @r###" | ||||
|             { | ||||
|               "message": "Index `test`: Attribute `doggos` is not filterable. Available filterable attribute patterns are: `doggos.*`.\n1:7 doggos EXISTS", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
| @@ -1,9 +1,12 @@ | ||||
| use meili_snap::{json_string, snapshot}; | ||||
| use meilisearch_types::milli::constants::RESERVED_GEO_FIELD_NAME; | ||||
| use once_cell::sync::Lazy; | ||||
|  | ||||
| use crate::common::{Server, Value}; | ||||
| use crate::json; | ||||
|  | ||||
| use super::test_settings_documents_indexing_swapping_and_search; | ||||
|  | ||||
| static DOCUMENTS: Lazy<Value> = Lazy::new(|| { | ||||
|     json!([ | ||||
|         { | ||||
| @@ -184,3 +187,184 @@ async fn bug_4640() { | ||||
|         ) | ||||
|         .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn geo_asc_with_words() { | ||||
|     let documents = json!([ | ||||
|       { "id": 0, "doggo": "jean", RESERVED_GEO_FIELD_NAME: { "lat": 0, "lng": 0 } }, | ||||
|       { "id": 1, "doggo": "intel", RESERVED_GEO_FIELD_NAME: { "lat": 88, "lng": 0 } }, | ||||
|       { "id": 2, "doggo": "jean bob", RESERVED_GEO_FIELD_NAME: { "lat": -89, "lng": 0 } }, | ||||
|       { "id": 3, "doggo": "jean michel", RESERVED_GEO_FIELD_NAME: { "lat": 0, "lng": 178 } }, | ||||
|       { "id": 4, "doggo": "bob marley", RESERVED_GEO_FIELD_NAME: { "lat": 0, "lng": -179 } }, | ||||
|     ]); | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &documents, | ||||
|         &json!({"searchableAttributes": ["id", "doggo"], "rankingRules": ["words", "geo:asc"]}), | ||||
|         &json!({"q": "jean"}), | ||||
|         |response, code| { | ||||
|             assert_eq!(code, 200, "{}", response); | ||||
|             snapshot!(json_string!(response, { ".processingTimeMs" => "[time]" }), @r###" | ||||
|             { | ||||
|               "hits": [ | ||||
|                 { | ||||
|                   "id": 0, | ||||
|                   "doggo": "jean", | ||||
|                   "_geo": { | ||||
|                     "lat": 0, | ||||
|                     "lng": 0 | ||||
|                   } | ||||
|                 }, | ||||
|                 { | ||||
|                   "id": 2, | ||||
|                   "doggo": "jean bob", | ||||
|                   "_geo": { | ||||
|                     "lat": -89, | ||||
|                     "lng": 0 | ||||
|                   } | ||||
|                 }, | ||||
|                 { | ||||
|                   "id": 3, | ||||
|                   "doggo": "jean michel", | ||||
|                   "_geo": { | ||||
|                     "lat": 0, | ||||
|                     "lng": 178 | ||||
|                   } | ||||
|                 } | ||||
|               ], | ||||
|               "query": "jean", | ||||
|               "processingTimeMs": "[time]", | ||||
|               "limit": 20, | ||||
|               "offset": 0, | ||||
|               "estimatedTotalHits": 3 | ||||
|             } | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &documents, | ||||
|         &json!({"searchableAttributes": ["id", "doggo"], "rankingRules": ["words", "geo:asc"]}), | ||||
|         &json!({"q": "bob"}), | ||||
|         |response, code| { | ||||
|             assert_eq!(code, 200, "{}", response); | ||||
|             snapshot!(json_string!(response, { ".processingTimeMs" => "[time]" }), @r###" | ||||
|             { | ||||
|               "hits": [ | ||||
|                 { | ||||
|                   "id": 2, | ||||
|                   "doggo": "jean bob", | ||||
|                   "_geo": { | ||||
|                     "lat": -89, | ||||
|                     "lng": 0 | ||||
|                   } | ||||
|                 }, | ||||
|                 { | ||||
|                   "id": 4, | ||||
|                   "doggo": "bob marley", | ||||
|                   "_geo": { | ||||
|                     "lat": 0, | ||||
|                     "lng": -179 | ||||
|                   } | ||||
|                 } | ||||
|               ], | ||||
|               "query": "bob", | ||||
|               "processingTimeMs": "[time]", | ||||
|               "limit": 20, | ||||
|               "offset": 0, | ||||
|               "estimatedTotalHits": 2 | ||||
|             } | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &documents, | ||||
|         &json!({"searchableAttributes": ["id", "doggo"], "rankingRules": ["words", "geo:asc"]}), | ||||
|         &json!({"q": "intel"}), | ||||
|         |response, code| { | ||||
|             assert_eq!(code, 200, "{}", response); | ||||
|             snapshot!(json_string!(response, { ".processingTimeMs" => "[time]" }), @r###" | ||||
|             { | ||||
|               "hits": [ | ||||
|                 { | ||||
|                   "id": 1, | ||||
|                   "doggo": "intel", | ||||
|                   "_geo": { | ||||
|                     "lat": 88, | ||||
|                     "lng": 0 | ||||
|                   } | ||||
|                 } | ||||
|               ], | ||||
|               "query": "intel", | ||||
|               "processingTimeMs": "[time]", | ||||
|               "limit": 20, | ||||
|               "offset": 0, | ||||
|               "estimatedTotalHits": 1 | ||||
|             } | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn geo_sort_with_words() { | ||||
|     let documents = json!([ | ||||
|       { "id": 0, "doggo": "jean", RESERVED_GEO_FIELD_NAME: { "lat": 0, "lng": 0 } }, | ||||
|       { "id": 1, "doggo": "intel", RESERVED_GEO_FIELD_NAME: { "lat": 88, "lng": 0 } }, | ||||
|       { "id": 2, "doggo": "jean bob", RESERVED_GEO_FIELD_NAME: { "lat": -89, "lng": 0 } }, | ||||
|       { "id": 3, "doggo": "jean michel", RESERVED_GEO_FIELD_NAME: { "lat": 0, "lng": 178 } }, | ||||
|       { "id": 4, "doggo": "bob marley", RESERVED_GEO_FIELD_NAME: { "lat": 0, "lng": -179 } }, | ||||
|     ]); | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|       &documents, | ||||
|       &json!({"searchableAttributes": ["id", "doggo"], "rankingRules": ["words", "sort"], "sortableAttributes": [RESERVED_GEO_FIELD_NAME]}), | ||||
|       &json!({"q": "jean", "sort": ["_geoPoint(0.0, 0.0):asc"]}), | ||||
|       |response, code| { | ||||
|           assert_eq!(code, 200, "{}", response); | ||||
|           snapshot!(json_string!(response, { ".processingTimeMs" => "[time]" }), @r###" | ||||
|           { | ||||
|             "hits": [ | ||||
|               { | ||||
|                 "id": 0, | ||||
|                 "doggo": "jean", | ||||
|                 "_geo": { | ||||
|                   "lat": 0, | ||||
|                   "lng": 0 | ||||
|                 }, | ||||
|                 "_geoDistance": 0 | ||||
|               }, | ||||
|               { | ||||
|                 "id": 2, | ||||
|                 "doggo": "jean bob", | ||||
|                 "_geo": { | ||||
|                   "lat": -89, | ||||
|                   "lng": 0 | ||||
|                 }, | ||||
|                 "_geoDistance": 9896348 | ||||
|               }, | ||||
|               { | ||||
|                 "id": 3, | ||||
|                 "doggo": "jean michel", | ||||
|                 "_geo": { | ||||
|                   "lat": 0, | ||||
|                   "lng": 178 | ||||
|                 }, | ||||
|                 "_geoDistance": 19792697 | ||||
|               } | ||||
|             ], | ||||
|             "query": "jean", | ||||
|             "processingTimeMs": "[time]", | ||||
|             "limit": 20, | ||||
|             "offset": 0, | ||||
|             "estimatedTotalHits": 3 | ||||
|           } | ||||
|           "###); | ||||
|       }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|   | ||||
| @@ -4,6 +4,7 @@ | ||||
| mod distinct; | ||||
| mod errors; | ||||
| mod facet_search; | ||||
| mod filters; | ||||
| mod formatted; | ||||
| mod geo; | ||||
| mod hybrid; | ||||
| @@ -21,10 +22,58 @@ use tempfile::TempDir; | ||||
|  | ||||
| use crate::common::{ | ||||
|     default_settings, shared_index_with_documents, shared_index_with_nested_documents, Server, | ||||
|     DOCUMENTS, FRUITS_DOCUMENTS, NESTED_DOCUMENTS, SCORE_DOCUMENTS, VECTOR_DOCUMENTS, | ||||
|     Value, DOCUMENTS, FRUITS_DOCUMENTS, NESTED_DOCUMENTS, SCORE_DOCUMENTS, VECTOR_DOCUMENTS, | ||||
| }; | ||||
| use crate::json; | ||||
|  | ||||
| async fn test_settings_documents_indexing_swapping_and_search( | ||||
|     documents: &Value, | ||||
|     settings: &Value, | ||||
|     query: &Value, | ||||
|     test: impl Fn(Value, actix_http::StatusCode) + std::panic::UnwindSafe + Clone, | ||||
| ) { | ||||
|     let temp = TempDir::new().unwrap(); | ||||
|     let server = Server::new_with_options(Opt { ..default_settings(temp.path()) }).await.unwrap(); | ||||
|  | ||||
|     eprintln!("Documents -> Settings -> test"); | ||||
|     let index = server.index("test"); | ||||
|  | ||||
|     let (task, code) = index.add_documents(documents.clone(), None).await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = index.wait_task(task.uid()).await; | ||||
|     assert!(response.is_success(), "{:?}", response); | ||||
|  | ||||
|     let (task, code) = index.update_settings(settings.clone()).await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = index.wait_task(task.uid()).await; | ||||
|     assert!(response.is_success(), "{:?}", response); | ||||
|  | ||||
|     index.search(query.clone(), test.clone()).await; | ||||
|     let (task, code) = server.delete_index("test").await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = server.wait_task(task.uid()).await; | ||||
|     assert!(response.is_success(), "{:?}", response); | ||||
|  | ||||
|     eprintln!("Settings -> Documents -> test"); | ||||
|     let index = server.index("test"); | ||||
|  | ||||
|     let (task, code) = index.update_settings(settings.clone()).await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = index.wait_task(task.uid()).await; | ||||
|     assert!(response.is_success(), "{:?}", response); | ||||
|  | ||||
|     let (task, code) = index.add_documents(documents.clone(), None).await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = index.wait_task(task.uid()).await; | ||||
|     assert!(response.is_success(), "{:?}", response); | ||||
|  | ||||
|     index.search(query.clone(), test.clone()).await; | ||||
|     let (task, code) = server.delete_index("test").await; | ||||
|     assert_eq!(code, 202, "{}", task); | ||||
|     let response = server.wait_task(task.uid()).await; | ||||
|     assert!(response.is_success(), "{:?}", response); | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn simple_placeholder_search() { | ||||
|     let index = shared_index_with_documents().await; | ||||
| @@ -355,118 +404,6 @@ async fn search_multiple_params() { | ||||
|         .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn search_with_filter_string_notation() { | ||||
|     let server = Server::new().await; | ||||
|     let index = server.index("test"); | ||||
|  | ||||
|     let (_, code) = index.update_settings(json!({"filterableAttributes": ["title"]})).await; | ||||
|     meili_snap::snapshot!(code, @"202 Accepted"); | ||||
|  | ||||
|     let documents = DOCUMENTS.clone(); | ||||
|     let (task, code) = index.add_documents(documents, None).await; | ||||
|     meili_snap::snapshot!(code, @"202 Accepted"); | ||||
|     let res = index.wait_task(task.uid()).await; | ||||
|     meili_snap::snapshot!(res["status"], @r###""succeeded""###); | ||||
|  | ||||
|     index | ||||
|         .search( | ||||
|             json!({ | ||||
|                 "filter": "title = Gläss" | ||||
|             }), | ||||
|             |response, code| { | ||||
|                 assert_eq!(code, 200, "{}", response); | ||||
|                 assert_eq!(response["hits"].as_array().unwrap().len(), 1); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
|  | ||||
|     let index = server.index("nested"); | ||||
|  | ||||
|     let (_, code) = | ||||
|         index.update_settings(json!({"filterableAttributes": ["cattos", "doggos.age"]})).await; | ||||
|     meili_snap::snapshot!(code, @"202 Accepted"); | ||||
|  | ||||
|     let documents = NESTED_DOCUMENTS.clone(); | ||||
|     let (task, code) = index.add_documents(documents, None).await; | ||||
|     meili_snap::snapshot!(code, @"202 Accepted"); | ||||
|     let res = index.wait_task(task.uid()).await; | ||||
|     meili_snap::snapshot!(res["status"], @r###""succeeded""###); | ||||
|  | ||||
|     index | ||||
|         .search( | ||||
|             json!({ | ||||
|                 "filter": "cattos = pésti" | ||||
|             }), | ||||
|             |response, code| { | ||||
|                 assert_eq!(code, 200, "{}", response); | ||||
|                 assert_eq!(response["hits"].as_array().unwrap().len(), 1); | ||||
|                 assert_eq!(response["hits"][0]["id"], json!(852)); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
|  | ||||
|     index | ||||
|         .search( | ||||
|             json!({ | ||||
|                 "filter": "doggos.age > 5" | ||||
|             }), | ||||
|             |response, code| { | ||||
|                 assert_eq!(code, 200, "{}", response); | ||||
|                 assert_eq!(response["hits"].as_array().unwrap().len(), 2); | ||||
|                 assert_eq!(response["hits"][0]["id"], json!(654)); | ||||
|                 assert_eq!(response["hits"][1]["id"], json!(951)); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn search_with_filter_array_notation() { | ||||
|     let index = shared_index_with_documents().await; | ||||
|     let (response, code) = index | ||||
|         .search_post(json!({ | ||||
|             "filter": ["title = Gläss"] | ||||
|         })) | ||||
|         .await; | ||||
|     assert_eq!(code, 200, "{}", response); | ||||
|     assert_eq!(response["hits"].as_array().unwrap().len(), 1); | ||||
|  | ||||
|     let (response, code) = index | ||||
|         .search_post(json!({ | ||||
|             "filter": [["title = Gläss", "title = \"Shazam!\"", "title = \"Escape Room\""]] | ||||
|         })) | ||||
|         .await; | ||||
|     assert_eq!(code, 200, "{}", response); | ||||
|     assert_eq!(response["hits"].as_array().unwrap().len(), 3); | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn search_with_contains_filter() { | ||||
|     let temp = TempDir::new().unwrap(); | ||||
|     let server = Server::new_with_options(Opt { | ||||
|         experimental_contains_filter: true, | ||||
|         ..default_settings(temp.path()) | ||||
|     }) | ||||
|     .await | ||||
|     .unwrap(); | ||||
|     let index = server.index("movies"); | ||||
|  | ||||
|     index.update_settings(json!({"filterableAttributes": ["title"]})).await; | ||||
|  | ||||
|     let documents = DOCUMENTS.clone(); | ||||
|     let (request, _code) = index.add_documents(documents, None).await; | ||||
|     index.wait_task(request.uid()).await.succeeded(); | ||||
|  | ||||
|     let (response, code) = index | ||||
|         .search_post(json!({ | ||||
|             "filter": "title CONTAINS cap" | ||||
|         })) | ||||
|         .await; | ||||
|     assert_eq!(code, 200, "{}", response); | ||||
|     assert_eq!(response["hits"].as_array().unwrap().len(), 2); | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn search_with_sort_on_numbers() { | ||||
|     let index = shared_index_with_documents().await; | ||||
| @@ -589,7 +526,7 @@ async fn search_facet_distribution() { | ||||
|             |response, code| { | ||||
|                 assert_eq!(code, 200, "{}", response); | ||||
|                 let dist = response["facetDistribution"].as_object().unwrap(); | ||||
|                 assert_eq!(dist.len(), 1); | ||||
|                 assert_eq!(dist.len(), 1, "{:?}", dist); | ||||
|                 assert_eq!( | ||||
|                     dist["doggos.name"], | ||||
|                     json!({ "bobby": 1, "buddy": 1, "gros bill": 1, "turbo": 1, "fast": 1}) | ||||
| @@ -606,7 +543,7 @@ async fn search_facet_distribution() { | ||||
|             |response, code| { | ||||
|                 assert_eq!(code, 200, "{}", response); | ||||
|                 let dist = response["facetDistribution"].as_object().unwrap(); | ||||
|                 assert_eq!(dist.len(), 3); | ||||
|                 assert_eq!(dist.len(), 3, "{:?}", dist); | ||||
|                 assert_eq!( | ||||
|                     dist["doggos.name"], | ||||
|                     json!({ "bobby": 1, "buddy": 1, "gros bill": 1, "turbo": 1, "fast": 1}) | ||||
| @@ -1559,6 +1496,293 @@ async fn change_attributes_settings() { | ||||
|         .await; | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn test_nested_fields() { | ||||
|     let documents = json!([ | ||||
|         { | ||||
|             "id": 0, | ||||
|             "title": "The zeroth document", | ||||
|         }, | ||||
|         { | ||||
|             "id": 1, | ||||
|             "title": "The first document", | ||||
|             "nested": { | ||||
|                 "object": "field", | ||||
|                 "machin": "bidule", | ||||
|             }, | ||||
|         }, | ||||
|         { | ||||
|             "id": 2, | ||||
|             "title": "The second document", | ||||
|             "nested": [ | ||||
|                 "array", | ||||
|                 { | ||||
|                     "object": "field", | ||||
|                 }, | ||||
|                 { | ||||
|                     "prout": "truc", | ||||
|                     "machin": "lol", | ||||
|                 }, | ||||
|             ], | ||||
|         }, | ||||
|         { | ||||
|             "id": 3, | ||||
|             "title": "The third document", | ||||
|             "nested": "I lied", | ||||
|         }, | ||||
|     ]); | ||||
|  | ||||
|     let settings = json!({ | ||||
|         "searchableAttributes": ["title", "nested.object", "nested.machin"], | ||||
|         "filterableAttributes": ["title", "nested.object", "nested.machin"] | ||||
|     }); | ||||
|  | ||||
|     // Test empty search returns all documents | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &documents, | ||||
|         &settings, | ||||
|         &json!({"q": "document"}), | ||||
|         |response, code| { | ||||
|             assert_eq!(code, 200, "{}", response); | ||||
|             snapshot!(json_string!(response["hits"]), @r###" | ||||
|         [ | ||||
|           { | ||||
|             "id": 0, | ||||
|             "title": "The zeroth document" | ||||
|           }, | ||||
|           { | ||||
|             "id": 1, | ||||
|             "title": "The first document", | ||||
|             "nested": { | ||||
|               "object": "field", | ||||
|               "machin": "bidule" | ||||
|             } | ||||
|           }, | ||||
|           { | ||||
|             "id": 2, | ||||
|             "title": "The second document", | ||||
|             "nested": [ | ||||
|               "array", | ||||
|               { | ||||
|                 "object": "field" | ||||
|               }, | ||||
|               { | ||||
|                 "prout": "truc", | ||||
|                 "machin": "lol" | ||||
|               } | ||||
|             ] | ||||
|           }, | ||||
|           { | ||||
|             "id": 3, | ||||
|             "title": "The third document", | ||||
|             "nested": "I lied" | ||||
|           } | ||||
|         ] | ||||
|         "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     // Test searching specific documents | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &documents, | ||||
|         &settings, | ||||
|         &json!({"q": "zeroth"}), | ||||
|         |response, code| { | ||||
|             assert_eq!(code, 200, "{}", response); | ||||
|             snapshot!(json_string!(response["hits"]), @r###" | ||||
|         [ | ||||
|           { | ||||
|             "id": 0, | ||||
|             "title": "The zeroth document" | ||||
|           } | ||||
|         ] | ||||
|         "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &documents, | ||||
|         &settings, | ||||
|         &json!({"q": "first"}), | ||||
|         |response, code| { | ||||
|             assert_eq!(code, 200, "{}", response); | ||||
|             snapshot!(json_string!(response["hits"]), @r###" | ||||
|         [ | ||||
|           { | ||||
|             "id": 1, | ||||
|             "title": "The first document", | ||||
|             "nested": { | ||||
|               "object": "field", | ||||
|               "machin": "bidule" | ||||
|             } | ||||
|           } | ||||
|         ] | ||||
|         "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     // Test searching nested fields | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &documents, | ||||
|         &settings, | ||||
|         &json!({"q": "field"}), | ||||
|         |response, code| { | ||||
|             assert_eq!(code, 200, "{}", response); | ||||
|             snapshot!(json_string!(response["hits"]), @r###" | ||||
|       [ | ||||
|         { | ||||
|           "id": 1, | ||||
|           "title": "The first document", | ||||
|           "nested": { | ||||
|             "object": "field", | ||||
|             "machin": "bidule" | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           "id": 2, | ||||
|           "title": "The second document", | ||||
|           "nested": [ | ||||
|             "array", | ||||
|             { | ||||
|               "object": "field" | ||||
|             }, | ||||
|             { | ||||
|               "prout": "truc", | ||||
|               "machin": "lol" | ||||
|             } | ||||
|           ] | ||||
|         } | ||||
|       ] | ||||
|       "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &documents, | ||||
|         &settings, | ||||
|         &json!({"q": "array"}), | ||||
|         |response, code| { | ||||
|             assert_eq!(code, 200, "{}", response); | ||||
|             // nested is not searchable | ||||
|             snapshot!(json_string!(response["hits"]), @"[]"); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &documents, | ||||
|         &settings, | ||||
|         &json!({"q": "lied"}), | ||||
|         |response, code| { | ||||
|             assert_eq!(code, 200, "{}", response); | ||||
|             // nested is not searchable | ||||
|             snapshot!(json_string!(response["hits"]), @"[]"); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     // Test filtering on nested fields | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &documents, | ||||
|         &settings, | ||||
|         &json!({"filter": "nested.object = field"}), | ||||
|         |response, code| { | ||||
|             assert_eq!(code, 200, "{}", response); | ||||
|             snapshot!(json_string!(response["hits"]), @r###" | ||||
|         [ | ||||
|           { | ||||
|             "id": 1, | ||||
|             "title": "The first document", | ||||
|             "nested": { | ||||
|               "object": "field", | ||||
|               "machin": "bidule" | ||||
|             } | ||||
|           }, | ||||
|           { | ||||
|             "id": 2, | ||||
|             "title": "The second document", | ||||
|             "nested": [ | ||||
|               "array", | ||||
|               { | ||||
|                 "object": "field" | ||||
|               }, | ||||
|               { | ||||
|                 "prout": "truc", | ||||
|                 "machin": "lol" | ||||
|               } | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|         "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &documents, | ||||
|         &settings, | ||||
|         &json!({"filter": "nested.machin = bidule"}), | ||||
|         |response, code| { | ||||
|             assert_eq!(code, 200, "{}", response); | ||||
|             snapshot!(json_string!(response["hits"]), @r###" | ||||
|         [ | ||||
|           { | ||||
|             "id": 1, | ||||
|             "title": "The first document", | ||||
|             "nested": { | ||||
|               "object": "field", | ||||
|               "machin": "bidule" | ||||
|             } | ||||
|           } | ||||
|         ] | ||||
|         "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     // Test filtering on non-filterable nested field fails | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &documents, | ||||
|         &settings, | ||||
|         &json!({"filter": "nested = array"}), | ||||
|         |response, code| { | ||||
|             assert_eq!(code, 400, "{}", response); | ||||
|             snapshot!(json_string!(response), @r###" | ||||
|             { | ||||
|               "message": "Index `test`: Attribute `nested` is not filterable. Available filterable attribute patterns are: `nested.machin`, `nested.object`, `title`.\n1:7 nested = array", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
|  | ||||
|     // Test filtering on non-filterable nested field fails | ||||
|     test_settings_documents_indexing_swapping_and_search( | ||||
|         &documents, | ||||
|         &settings, | ||||
|         &json!({"filter": r#"nested = "I lied""#}), | ||||
|         |response, code| { | ||||
|             assert_eq!(code, 400, "{}", response); | ||||
|             snapshot!(json_string!(response), @r###" | ||||
|             { | ||||
|               "message": "Index `test`: Attribute `nested` is not filterable. Available filterable attribute patterns are: `nested.machin`, `nested.object`, `title`.\n1:7 nested = \"I lied\"", | ||||
|               "code": "invalid_search_filter", | ||||
|               "type": "invalid_request", | ||||
|               "link": "https://docs.meilisearch.com/errors#invalid_search_filter" | ||||
|             } | ||||
|             "###); | ||||
|         }, | ||||
|     ) | ||||
|     .await; | ||||
| } | ||||
|  | ||||
| /// Modifying facets with different casing should work correctly | ||||
| #[actix_rt::test] | ||||
| async fn change_facet_casing() { | ||||
|   | ||||
| @@ -3647,7 +3647,7 @@ async fn federation_non_faceted_for_an_index() { | ||||
|     snapshot!(code, @"400 Bad Request"); | ||||
|     insta::assert_json_snapshot!(response, { ".processingTimeMs" => "[time]" }, @r###" | ||||
|     { | ||||
|       "message": "Inside `.federation.facetsByIndex.fruits-no-name`: Invalid facet distribution, attribute `name` is not filterable. The available filterable attributes are `BOOST, id`.\n - Note: index `fruits-no-name` used in `.queries[1]`", | ||||
|       "message": "Inside `.federation.facetsByIndex.fruits-no-name`: Invalid facet distribution, attribute `name` is not filterable. The available filterable attribute patterns are `BOOST, id`.\n - Note: index `fruits-no-name` used in `.queries[1]`", | ||||
|       "code": "invalid_multi_search_facets", | ||||
|       "type": "invalid_request", | ||||
|       "link": "https://docs.meilisearch.com/errors#invalid_multi_search_facets" | ||||
| @@ -3669,7 +3669,7 @@ async fn federation_non_faceted_for_an_index() { | ||||
|     snapshot!(code, @"400 Bad Request"); | ||||
|     insta::assert_json_snapshot!(response, { ".processingTimeMs" => "[time]" }, @r###" | ||||
|     { | ||||
|       "message": "Inside `.federation.facetsByIndex.fruits-no-name`: Invalid facet distribution, attribute `name` is not filterable. The available filterable attributes are `BOOST, id`.\n - Note: index `fruits-no-name` is not used in queries", | ||||
|       "message": "Inside `.federation.facetsByIndex.fruits-no-name`: Invalid facet distribution, attribute `name` is not filterable. The available filterable attribute patterns are `BOOST, id`.\n - Note: index `fruits-no-name` is not used in queries", | ||||
|       "code": "invalid_multi_search_facets", | ||||
|       "type": "invalid_request", | ||||
|       "link": "https://docs.meilisearch.com/errors#invalid_multi_search_facets" | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| use meili_snap::{json_string, snapshot}; | ||||
|  | ||||
| use crate::common::Server; | ||||
| use crate::json; | ||||
|  | ||||
| @@ -510,3 +512,127 @@ async fn set_and_reset_distinct_attribute_with_dedicated_route() { | ||||
|  | ||||
|     assert_eq!(response, json!(null)); | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn granular_filterable_attributes() { | ||||
|     let server = Server::new().await; | ||||
|     let index = server.index("test"); | ||||
|     index.create(None).await; | ||||
|  | ||||
|     let (response, code) = | ||||
|         index.update_settings(json!({ "filterableAttributes": [ | ||||
|             { "attributePatterns": ["name"], "features": { "facetSearch": true, "filter": {"equality": true, "comparison": false} } }, | ||||
|             { "attributePatterns": ["age"], "features": { "facetSearch": false, "filter": {"equality": true, "comparison": true} } }, | ||||
|             { "attributePatterns": ["id"] }, | ||||
|             { "attributePatterns": ["default-filterable-features-null"], "features": { "facetSearch": true } }, | ||||
|             { "attributePatterns": ["default-filterable-features-equality"], "features": { "facetSearch": true, "filter": {"comparison": true} } }, | ||||
|             { "attributePatterns": ["default-filterable-features-comparison"], "features": { "facetSearch": true, "filter": {"equality": true} } }, | ||||
|             { "attributePatterns": ["default-filterable-features-empty"], "features": { "facetSearch": true, "filter": {} } }, | ||||
|             { "attributePatterns": ["default-facet-search"], "features": { "filter": {"equality": true, "comparison": true} } }, | ||||
|         ] })).await; | ||||
|     assert_eq!(code, 202); | ||||
|     index.wait_task(response.uid()).await.succeeded(); | ||||
|  | ||||
|     let (response, code) = index.settings().await; | ||||
|     assert_eq!(code, 200, "{}", response); | ||||
|     snapshot!(json_string!(response["filterableAttributes"]), @r###" | ||||
|     [ | ||||
|       { | ||||
|         "attributePatterns": [ | ||||
|           "name" | ||||
|         ], | ||||
|         "features": { | ||||
|           "facetSearch": true, | ||||
|           "filter": { | ||||
|             "equality": true, | ||||
|             "comparison": false | ||||
|           } | ||||
|         } | ||||
|       }, | ||||
|       { | ||||
|         "attributePatterns": [ | ||||
|           "age" | ||||
|         ], | ||||
|         "features": { | ||||
|           "facetSearch": false, | ||||
|           "filter": { | ||||
|             "equality": true, | ||||
|             "comparison": true | ||||
|           } | ||||
|         } | ||||
|       }, | ||||
|       { | ||||
|         "attributePatterns": [ | ||||
|           "id" | ||||
|         ], | ||||
|         "features": { | ||||
|           "facetSearch": false, | ||||
|           "filter": { | ||||
|             "equality": true, | ||||
|             "comparison": false | ||||
|           } | ||||
|         } | ||||
|       }, | ||||
|       { | ||||
|         "attributePatterns": [ | ||||
|           "default-filterable-features-null" | ||||
|         ], | ||||
|         "features": { | ||||
|           "facetSearch": true, | ||||
|           "filter": { | ||||
|             "equality": true, | ||||
|             "comparison": false | ||||
|           } | ||||
|         } | ||||
|       }, | ||||
|       { | ||||
|         "attributePatterns": [ | ||||
|           "default-filterable-features-equality" | ||||
|         ], | ||||
|         "features": { | ||||
|           "facetSearch": true, | ||||
|           "filter": { | ||||
|             "equality": true, | ||||
|             "comparison": true | ||||
|           } | ||||
|         } | ||||
|       }, | ||||
|       { | ||||
|         "attributePatterns": [ | ||||
|           "default-filterable-features-comparison" | ||||
|         ], | ||||
|         "features": { | ||||
|           "facetSearch": true, | ||||
|           "filter": { | ||||
|             "equality": true, | ||||
|             "comparison": false | ||||
|           } | ||||
|         } | ||||
|       }, | ||||
|       { | ||||
|         "attributePatterns": [ | ||||
|           "default-filterable-features-empty" | ||||
|         ], | ||||
|         "features": { | ||||
|           "facetSearch": true, | ||||
|           "filter": { | ||||
|             "equality": true, | ||||
|             "comparison": false | ||||
|           } | ||||
|         } | ||||
|       }, | ||||
|       { | ||||
|         "attributePatterns": [ | ||||
|           "default-facet-search" | ||||
|         ], | ||||
|         "features": { | ||||
|           "facetSearch": false, | ||||
|           "filter": { | ||||
|             "equality": true, | ||||
|             "comparison": true | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     ] | ||||
|     "###); | ||||
| } | ||||
|   | ||||
| @@ -452,18 +452,19 @@ async fn filter_invalid_attribute_array() { | ||||
|     snapshot!(code, @"202 Accepted"); | ||||
|     index.wait_task(value.uid()).await.succeeded(); | ||||
|  | ||||
|     let expected_response = json!({ | ||||
|         "message": "Attribute `many` is not filterable. Available filterable attributes are: `title`.\n1:5 many = Glass", | ||||
|         "code": "invalid_similar_filter", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_similar_filter" | ||||
|     }); | ||||
|     index | ||||
|         .similar( | ||||
|             json!({"id": 287947, "filter": ["many = Glass"], "embedder": "manual"}), | ||||
|             |response, code| { | ||||
|                 assert_eq!(response, expected_response); | ||||
|                 assert_eq!(code, 400); | ||||
|                 snapshot!(code, @"400 Bad Request"); | ||||
|                 snapshot!(response, @r###" | ||||
|                 { | ||||
|                   "message": "Attribute `many` is not filterable. Available filterable attribute patterns are: `title`.\n1:5 many = Glass", | ||||
|                   "code": "invalid_similar_filter", | ||||
|                   "type": "invalid_request", | ||||
|                   "link": "https://docs.meilisearch.com/errors#invalid_similar_filter" | ||||
|                 } | ||||
|                 "###); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
| @@ -492,18 +493,19 @@ async fn filter_invalid_attribute_string() { | ||||
|     snapshot!(code, @"202 Accepted"); | ||||
|     index.wait_task(value.uid()).await.succeeded(); | ||||
|  | ||||
|     let expected_response = json!({ | ||||
|         "message": "Attribute `many` is not filterable. Available filterable attributes are: `title`.\n1:5 many = Glass", | ||||
|         "code": "invalid_similar_filter", | ||||
|         "type": "invalid_request", | ||||
|         "link": "https://docs.meilisearch.com/errors#invalid_similar_filter" | ||||
|     }); | ||||
|     index | ||||
|         .similar( | ||||
|             json!({"id": 287947, "filter": "many = Glass", "embedder": "manual"}), | ||||
|             |response, code| { | ||||
|                 assert_eq!(response, expected_response); | ||||
|                 assert_eq!(code, 400); | ||||
|                 snapshot!(code, @"400 Bad Request"); | ||||
|                 snapshot!(response, @r###" | ||||
|                 { | ||||
|                   "message": "Attribute `many` is not filterable. Available filterable attribute patterns are: `title`.\n1:5 many = Glass", | ||||
|                   "code": "invalid_similar_filter", | ||||
|                   "type": "invalid_request", | ||||
|                   "link": "https://docs.meilisearch.com/errors#invalid_similar_filter" | ||||
|                 } | ||||
|                 "###); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
|   | ||||
		Reference in New Issue
	
	Block a user