mirror of
https://github.com/meilisearch/meilisearch.git
synced 2024-11-27 04:25:06 +08:00
Merge #3915
3915: `attributesToSearchOn` supports wildcards r=ManyTheFish a=dureuill # Pull Request ## Related issue Fixes #3912 and #3911 ## What does this PR do? - Adding `*` in the list of `attributesToSearchOn` allows searching on all the `searchableAttributes`. - If `searchableAttributes contains "*"`, then any attribute is accepted in the `attributesToSearchOn` list. Co-authored-by: Louis Dureuil <louis@meilisearch.com>
This commit is contained in:
commit
fd7c66fd62
@ -968,9 +968,12 @@ async fn sort_unset_ranking_rule() {
|
|||||||
async fn search_on_unknown_field() {
|
async fn search_on_unknown_field() {
|
||||||
let server = Server::new().await;
|
let server = Server::new().await;
|
||||||
let index = server.index("test");
|
let index = server.index("test");
|
||||||
|
index.update_settings_searchable_attributes(json!(["id", "title"])).await;
|
||||||
|
index.wait_task(0).await;
|
||||||
|
|
||||||
let documents = DOCUMENTS.clone();
|
let documents = DOCUMENTS.clone();
|
||||||
index.add_documents(documents, None).await;
|
index.add_documents(documents, None).await;
|
||||||
index.wait_task(0).await;
|
index.wait_task(1).await;
|
||||||
|
|
||||||
index
|
index
|
||||||
.search(
|
.search(
|
||||||
@ -989,3 +992,49 @@ async fn search_on_unknown_field() {
|
|||||||
)
|
)
|
||||||
.await;
|
.await;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[actix_rt::test]
|
||||||
|
async fn search_on_unknown_field_plus_joker() {
|
||||||
|
let server = Server::new().await;
|
||||||
|
let index = server.index("test");
|
||||||
|
index.update_settings_searchable_attributes(json!(["id", "title"])).await;
|
||||||
|
index.wait_task(0).await;
|
||||||
|
|
||||||
|
let documents = DOCUMENTS.clone();
|
||||||
|
index.add_documents(documents, None).await;
|
||||||
|
index.wait_task(1).await;
|
||||||
|
|
||||||
|
index
|
||||||
|
.search(
|
||||||
|
json!({"q": "Captain Marvel", "attributesToSearchOn": ["*", "unknown"]}),
|
||||||
|
|response, code| {
|
||||||
|
snapshot!(code, @"400 Bad Request");
|
||||||
|
snapshot!(json_string!(response), @r###"
|
||||||
|
{
|
||||||
|
"message": "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;
|
||||||
|
|
||||||
|
index
|
||||||
|
.search(
|
||||||
|
json!({"q": "Captain Marvel", "attributesToSearchOn": ["unknown", "*"]}),
|
||||||
|
|response, code| {
|
||||||
|
snapshot!(code, @"400 Bad Request");
|
||||||
|
snapshot!(json_string!(response), @r###"
|
||||||
|
{
|
||||||
|
"message": "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;
|
||||||
|
}
|
||||||
|
@ -49,6 +49,76 @@ async fn simple_search_on_title() {
|
|||||||
.await;
|
.await;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[actix_rt::test]
|
||||||
|
async fn search_no_searchable_attribute_set() {
|
||||||
|
let server = Server::new().await;
|
||||||
|
let index = index_with_documents(&server, &SIMPLE_SEARCH_DOCUMENTS).await;
|
||||||
|
|
||||||
|
index
|
||||||
|
.search(
|
||||||
|
json!({"q": "Captain Marvel", "attributesToSearchOn": ["unknown"]}),
|
||||||
|
|response, code| {
|
||||||
|
snapshot!(code, @"200 OK");
|
||||||
|
snapshot!(response["hits"].as_array().unwrap().len(), @"0");
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
|
||||||
|
index.update_settings_searchable_attributes(json!(["*"])).await;
|
||||||
|
index.wait_task(1).await;
|
||||||
|
|
||||||
|
index
|
||||||
|
.search(
|
||||||
|
json!({"q": "Captain Marvel", "attributesToSearchOn": ["unknown"]}),
|
||||||
|
|response, code| {
|
||||||
|
snapshot!(code, @"200 OK");
|
||||||
|
snapshot!(response["hits"].as_array().unwrap().len(), @"0");
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
|
||||||
|
index.update_settings_searchable_attributes(json!(["*"])).await;
|
||||||
|
index.wait_task(2).await;
|
||||||
|
|
||||||
|
index
|
||||||
|
.search(
|
||||||
|
json!({"q": "Captain Marvel", "attributesToSearchOn": ["unknown", "title"]}),
|
||||||
|
|response, code| {
|
||||||
|
snapshot!(code, @"200 OK");
|
||||||
|
snapshot!(response["hits"].as_array().unwrap().len(), @"2");
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
|
||||||
|
#[actix_rt::test]
|
||||||
|
async fn search_on_all_attributes() {
|
||||||
|
let server = Server::new().await;
|
||||||
|
let index = index_with_documents(&server, &SIMPLE_SEARCH_DOCUMENTS).await;
|
||||||
|
|
||||||
|
index
|
||||||
|
.search(json!({"q": "Captain Marvel", "attributesToSearchOn": ["*"]}), |response, code| {
|
||||||
|
snapshot!(code, @"200 OK");
|
||||||
|
snapshot!(response["hits"].as_array().unwrap().len(), @"3");
|
||||||
|
})
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
|
||||||
|
#[actix_rt::test]
|
||||||
|
async fn search_on_all_attributes_restricted_set() {
|
||||||
|
let server = Server::new().await;
|
||||||
|
let index = index_with_documents(&server, &SIMPLE_SEARCH_DOCUMENTS).await;
|
||||||
|
index.update_settings_searchable_attributes(json!(["title"])).await;
|
||||||
|
index.wait_task(1).await;
|
||||||
|
|
||||||
|
index
|
||||||
|
.search(json!({"q": "Captain Marvel", "attributesToSearchOn": ["*"]}), |response, code| {
|
||||||
|
snapshot!(code, @"200 OK");
|
||||||
|
snapshot!(response["hits"].as_array().unwrap().len(), @"2");
|
||||||
|
})
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
|
||||||
#[actix_rt::test]
|
#[actix_rt::test]
|
||||||
async fn simple_prefix_search_on_title() {
|
async fn simple_prefix_search_on_title() {
|
||||||
let server = Server::new().await;
|
let server = Server::new().await;
|
||||||
|
@ -85,7 +85,12 @@ impl<'ctx> SearchContext<'ctx> {
|
|||||||
let searchable_names = self.index.searchable_fields(self.txn)?;
|
let searchable_names = self.index.searchable_fields(self.txn)?;
|
||||||
|
|
||||||
let mut restricted_fids = Vec::new();
|
let mut restricted_fids = Vec::new();
|
||||||
|
let mut contains_wildcard = false;
|
||||||
for field_name in searchable_attributes {
|
for field_name in searchable_attributes {
|
||||||
|
if field_name == "*" {
|
||||||
|
contains_wildcard = true;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
let searchable_contains_name =
|
let searchable_contains_name =
|
||||||
searchable_names.as_ref().map(|sn| sn.iter().any(|name| name == field_name));
|
searchable_names.as_ref().map(|sn| sn.iter().any(|name| name == field_name));
|
||||||
let fid = match (fids_map.id(field_name), searchable_contains_name) {
|
let fid = match (fids_map.id(field_name), searchable_contains_name) {
|
||||||
@ -99,8 +104,10 @@ impl<'ctx> SearchContext<'ctx> {
|
|||||||
}
|
}
|
||||||
.into())
|
.into())
|
||||||
}
|
}
|
||||||
|
// The field is not searchable, but the searchableAttributes are set to * => ignore field
|
||||||
|
(None, None) => continue,
|
||||||
// The field is not searchable => User error
|
// The field is not searchable => User error
|
||||||
_otherwise => {
|
(_fid, Some(false)) => {
|
||||||
let mut valid_fields: BTreeSet<_> =
|
let mut valid_fields: BTreeSet<_> =
|
||||||
fids_map.names().map(String::from).collect();
|
fids_map.names().map(String::from).collect();
|
||||||
|
|
||||||
@ -132,7 +139,7 @@ impl<'ctx> SearchContext<'ctx> {
|
|||||||
restricted_fids.push(fid);
|
restricted_fids.push(fid);
|
||||||
}
|
}
|
||||||
|
|
||||||
self.restricted_fids = Some(restricted_fids);
|
self.restricted_fids = (!contains_wildcard).then_some(restricted_fids);
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user