mirror of
https://github.com/meilisearch/MeiliSearch
synced 2025-01-15 07:47:30 +01:00
798 lines
29 KiB
Rust
798 lines
29 KiB
Rust
use meili_snap::*;
|
|
use urlencoding::encode;
|
|
|
|
use crate::common::Server;
|
|
use crate::json;
|
|
|
|
#[actix_rt::test]
|
|
async fn get_all_documents_bad_offset() {
|
|
let server = Server::new().await;
|
|
let index = server.index("test");
|
|
|
|
let (response, code) = index.get_all_documents_raw("?offset").await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid value in parameter `offset`: could not parse `` as a positive integer",
|
|
"code": "invalid_document_offset",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_offset"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.get_all_documents_raw("?offset=doggo").await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid value in parameter `offset`: could not parse `doggo` as a positive integer",
|
|
"code": "invalid_document_offset",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_offset"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.get_all_documents_raw("?offset=-1").await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid value in parameter `offset`: could not parse `-1` as a positive integer",
|
|
"code": "invalid_document_offset",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_offset"
|
|
}
|
|
"###);
|
|
}
|
|
|
|
#[actix_rt::test]
|
|
async fn get_all_documents_bad_limit() {
|
|
let server = Server::new().await;
|
|
let index = server.index("test");
|
|
|
|
let (response, code) = index.get_all_documents_raw("?limit").await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid value in parameter `limit`: could not parse `` as a positive integer",
|
|
"code": "invalid_document_limit",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_limit"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.get_all_documents_raw("?limit=doggo").await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid value in parameter `limit`: could not parse `doggo` as a positive integer",
|
|
"code": "invalid_document_limit",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_limit"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.get_all_documents_raw("?limit=-1").await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid value in parameter `limit`: could not parse `-1` as a positive integer",
|
|
"code": "invalid_document_limit",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_limit"
|
|
}
|
|
"###);
|
|
}
|
|
|
|
#[actix_rt::test]
|
|
async fn get_all_documents_bad_filter() {
|
|
let server = Server::new().await;
|
|
let index = server.index("test");
|
|
|
|
// Since the filter can't be parsed automatically by deserr, we have the wrong error message
|
|
// if the index does not exist: we could expect to get an error message about the invalid filter before
|
|
// the existence of the index is checked, but it is not the case.
|
|
let (response, code) = index.get_all_documents_raw("?filter").await;
|
|
snapshot!(code, @"404 Not Found");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Index `test` not found.",
|
|
"code": "index_not_found",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#index_not_found"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.get_all_documents_raw("?filter=doggo").await;
|
|
snapshot!(code, @"404 Not Found");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Index `test` not found.",
|
|
"code": "index_not_found",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#index_not_found"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.get_all_documents_raw("?filter=doggo=bernese").await;
|
|
snapshot!(code, @"404 Not Found");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Index `test` not found.",
|
|
"code": "index_not_found",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#index_not_found"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.create(None).await;
|
|
snapshot!(code, @"202 Accepted");
|
|
snapshot!(json_string!(response, { ".enqueuedAt" => "[date]" }), @r###"
|
|
{
|
|
"taskUid": 0,
|
|
"indexUid": "test",
|
|
"status": "enqueued",
|
|
"type": "indexCreation",
|
|
"enqueuedAt": "[date]"
|
|
}
|
|
"###);
|
|
let response = server.wait_task(0).await;
|
|
snapshot!(json_string!(response, { ".duration" => "[duration]", ".enqueuedAt" => "[date]", ".startedAt" => "[date]", ".finishedAt" => "[date]" }), @r###"
|
|
{
|
|
"uid": 0,
|
|
"indexUid": "test",
|
|
"status": "succeeded",
|
|
"type": "indexCreation",
|
|
"canceledBy": null,
|
|
"details": {
|
|
"primaryKey": null
|
|
},
|
|
"error": null,
|
|
"duration": "[duration]",
|
|
"enqueuedAt": "[date]",
|
|
"startedAt": "[date]",
|
|
"finishedAt": "[date]"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.get_all_documents_raw("?filter").await;
|
|
snapshot!(code, @"200 OK");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"results": [],
|
|
"offset": 0,
|
|
"limit": 20,
|
|
"total": 0
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.get_all_documents_raw("?filter=doggo").await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` at `doggo`.\n1:6 doggo",
|
|
"code": "invalid_document_filter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_filter"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.get_all_documents_raw("?filter=doggo=bernese").await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Attribute `doggo` is not filterable. This index does not have configured filterable attributes.\n1:6 doggo=bernese",
|
|
"code": "invalid_document_filter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_filter"
|
|
}
|
|
"###);
|
|
}
|
|
|
|
#[actix_rt::test]
|
|
async fn delete_documents_batch() {
|
|
let server = Server::new().await;
|
|
let index = server.index("test");
|
|
|
|
let (response, code) = index.delete_batch_raw(json!("doggo")).await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Json deserialize error: invalid type: string \"doggo\", expected a sequence at line 1 column 7",
|
|
"code": "bad_request",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#bad_request"
|
|
}
|
|
"###);
|
|
}
|
|
|
|
#[actix_rt::test]
|
|
async fn replace_documents_missing_payload() {
|
|
let server = Server::new().await;
|
|
let index = server.index("test");
|
|
|
|
let (response, code) =
|
|
index.raw_add_documents("", vec![("Content-Type", "application/json")], "").await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "A json payload is missing.",
|
|
"code": "missing_payload",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#missing_payload"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) =
|
|
index.raw_add_documents("", vec![("Content-Type", "application/x-ndjson")], "").await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "A ndjson payload is missing.",
|
|
"code": "missing_payload",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#missing_payload"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) =
|
|
index.raw_add_documents("", vec![("Content-Type", "text/csv")], "").await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "A csv payload is missing.",
|
|
"code": "missing_payload",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#missing_payload"
|
|
}
|
|
"###);
|
|
}
|
|
|
|
#[actix_rt::test]
|
|
async fn update_documents_missing_payload() {
|
|
let server = Server::new().await;
|
|
let index = server.index("test");
|
|
|
|
let (response, code) = index.raw_update_documents("", Some("application/json"), "").await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "A json payload is missing.",
|
|
"code": "missing_payload",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#missing_payload"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.raw_update_documents("", Some("application/x-ndjson"), "").await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "A ndjson payload is missing.",
|
|
"code": "missing_payload",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#missing_payload"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.raw_update_documents("", Some("text/csv"), "").await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "A csv payload is missing.",
|
|
"code": "missing_payload",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#missing_payload"
|
|
}
|
|
"###);
|
|
}
|
|
|
|
#[actix_rt::test]
|
|
async fn replace_documents_missing_content_type() {
|
|
let server = Server::new().await;
|
|
let index = server.index("test");
|
|
|
|
let (response, code) = index.raw_add_documents("", Vec::new(), "").await;
|
|
snapshot!(code, @"415 Unsupported Media Type");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "A Content-Type header is missing. Accepted values for the Content-Type header are: `application/json`, `application/x-ndjson`, `text/csv`",
|
|
"code": "missing_content_type",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#missing_content_type"
|
|
}
|
|
"###);
|
|
|
|
// even with a csv delimiter specified this error is triggered first
|
|
let (response, code) = index.raw_add_documents("", Vec::new(), "?csvDelimiter=;").await;
|
|
snapshot!(code, @"415 Unsupported Media Type");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "A Content-Type header is missing. Accepted values for the Content-Type header are: `application/json`, `application/x-ndjson`, `text/csv`",
|
|
"code": "missing_content_type",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#missing_content_type"
|
|
}
|
|
"###);
|
|
}
|
|
|
|
#[actix_rt::test]
|
|
async fn update_documents_missing_content_type() {
|
|
let server = Server::new().await;
|
|
let index = server.index("test");
|
|
|
|
let (response, code) = index.raw_update_documents("", None, "").await;
|
|
snapshot!(code, @"415 Unsupported Media Type");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "A Content-Type header is missing. Accepted values for the Content-Type header are: `application/json`, `application/x-ndjson`, `text/csv`",
|
|
"code": "missing_content_type",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#missing_content_type"
|
|
}
|
|
"###);
|
|
|
|
// even with a csv delimiter specified this error is triggered first
|
|
let (response, code) = index.raw_update_documents("", None, "?csvDelimiter=;").await;
|
|
snapshot!(code, @"415 Unsupported Media Type");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "A Content-Type header is missing. Accepted values for the Content-Type header are: `application/json`, `application/x-ndjson`, `text/csv`",
|
|
"code": "missing_content_type",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#missing_content_type"
|
|
}
|
|
"###);
|
|
}
|
|
|
|
#[actix_rt::test]
|
|
async fn replace_documents_bad_content_type() {
|
|
let server = Server::new().await;
|
|
let index = server.index("test");
|
|
|
|
let (response, code) = index.raw_add_documents("", vec![("Content-Type", "doggo")], "").await;
|
|
snapshot!(code, @"415 Unsupported Media Type");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "The Content-Type `doggo` is invalid. Accepted values for the Content-Type header are: `application/json`, `application/x-ndjson`, `text/csv`",
|
|
"code": "invalid_content_type",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_content_type"
|
|
}
|
|
"###);
|
|
}
|
|
|
|
#[actix_rt::test]
|
|
async fn update_documents_bad_content_type() {
|
|
let server = Server::new().await;
|
|
let index = server.index("test");
|
|
|
|
let (response, code) = index.raw_update_documents("", Some("doggo"), "").await;
|
|
snapshot!(code, @"415 Unsupported Media Type");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "The Content-Type `doggo` is invalid. Accepted values for the Content-Type header are: `application/json`, `application/x-ndjson`, `text/csv`",
|
|
"code": "invalid_content_type",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_content_type"
|
|
}
|
|
"###);
|
|
}
|
|
|
|
#[actix_rt::test]
|
|
async fn replace_documents_bad_csv_delimiter() {
|
|
let server = Server::new().await;
|
|
let index = server.index("test");
|
|
|
|
let (response, code) = index
|
|
.raw_add_documents("", vec![("Content-Type", "application/json")], "?csvDelimiter")
|
|
.await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid value in parameter `csvDelimiter`: expected a string of one character, but found an empty string",
|
|
"code": "invalid_document_csv_delimiter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_csv_delimiter"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index
|
|
.raw_add_documents("", vec![("Content-Type", "application/json")], "?csvDelimiter=doggo")
|
|
.await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid value in parameter `csvDelimiter`: expected a string of one character, but found the following string of 5 characters: `doggo`",
|
|
"code": "invalid_document_csv_delimiter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_csv_delimiter"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index
|
|
.raw_add_documents(
|
|
"",
|
|
vec![("Content-Type", "application/json")],
|
|
&format!("?csvDelimiter={}", encode("🍰")),
|
|
)
|
|
.await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "csv delimiter must be an ascii character. Found: `🍰`",
|
|
"code": "invalid_document_csv_delimiter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_csv_delimiter"
|
|
}
|
|
"###);
|
|
}
|
|
|
|
#[actix_rt::test]
|
|
async fn update_documents_bad_csv_delimiter() {
|
|
let server = Server::new().await;
|
|
let index = server.index("test");
|
|
|
|
let (response, code) =
|
|
index.raw_update_documents("", Some("application/json"), "?csvDelimiter").await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid value in parameter `csvDelimiter`: expected a string of one character, but found an empty string",
|
|
"code": "invalid_document_csv_delimiter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_csv_delimiter"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) =
|
|
index.raw_update_documents("", Some("application/json"), "?csvDelimiter=doggo").await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid value in parameter `csvDelimiter`: expected a string of one character, but found the following string of 5 characters: `doggo`",
|
|
"code": "invalid_document_csv_delimiter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_csv_delimiter"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index
|
|
.raw_update_documents(
|
|
"",
|
|
Some("application/json"),
|
|
&format!("?csvDelimiter={}", encode("🍰")),
|
|
)
|
|
.await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "csv delimiter must be an ascii character. Found: `🍰`",
|
|
"code": "invalid_document_csv_delimiter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_csv_delimiter"
|
|
}
|
|
"###);
|
|
}
|
|
|
|
#[actix_rt::test]
|
|
async fn replace_documents_csv_delimiter_with_bad_content_type() {
|
|
let server = Server::new().await;
|
|
let index = server.index("test");
|
|
|
|
let (response, code) = index
|
|
.raw_add_documents("", vec![("Content-Type", "application/json")], "?csvDelimiter=a")
|
|
.await;
|
|
snapshot!(code, @"415 Unsupported Media Type");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "The Content-Type `application/json` does not support the use of a csv delimiter. The csv delimiter can only be used with the Content-Type `text/csv`.",
|
|
"code": "invalid_content_type",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_content_type"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index
|
|
.raw_add_documents("", vec![("Content-Type", "application/x-ndjson")], "?csvDelimiter=a")
|
|
.await;
|
|
snapshot!(code, @"415 Unsupported Media Type");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "The Content-Type `application/x-ndjson` does not support the use of a csv delimiter. The csv delimiter can only be used with the Content-Type `text/csv`.",
|
|
"code": "invalid_content_type",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_content_type"
|
|
}
|
|
"###);
|
|
}
|
|
|
|
#[actix_rt::test]
|
|
async fn update_documents_csv_delimiter_with_bad_content_type() {
|
|
let server = Server::new().await;
|
|
let index = server.index("test");
|
|
|
|
let (response, code) =
|
|
index.raw_update_documents("", Some("application/json"), "?csvDelimiter=a").await;
|
|
snapshot!(code, @"415 Unsupported Media Type");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "The Content-Type `application/json` does not support the use of a csv delimiter. The csv delimiter can only be used with the Content-Type `text/csv`.",
|
|
"code": "invalid_content_type",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_content_type"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) =
|
|
index.raw_update_documents("", Some("application/x-ndjson"), "?csvDelimiter=a").await;
|
|
snapshot!(code, @"415 Unsupported Media Type");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "The Content-Type `application/x-ndjson` does not support the use of a csv delimiter. The csv delimiter can only be used with the Content-Type `text/csv`.",
|
|
"code": "invalid_content_type",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_content_type"
|
|
}
|
|
"###);
|
|
}
|
|
|
|
#[actix_rt::test]
|
|
async fn delete_document_by_filter() {
|
|
let server = Server::new().await;
|
|
let index = server.index("doggo");
|
|
|
|
// send a bad payload type
|
|
let (response, code) = index.delete_document_by_filter(json!("hello")).await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid value type: expected an object, but found a string: `\"hello\"`",
|
|
"code": "bad_request",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#bad_request"
|
|
}
|
|
"###);
|
|
|
|
// send bad payload type
|
|
let (response, code) = index.delete_document_by_filter(json!({ "filter": true })).await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid syntax for the filter parameter: `expected String, Array, found: true`.",
|
|
"code": "invalid_document_filter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_filter"
|
|
}
|
|
"###);
|
|
|
|
// send bad filter
|
|
let (response, code) = index.delete_document_by_filter(json!({ "filter": "hello"})).await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` at `hello`.\n1:6 hello",
|
|
"code": "invalid_document_filter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_filter"
|
|
}
|
|
"###);
|
|
|
|
// send empty filter
|
|
let (response, code) = index.delete_document_by_filter(json!({ "filter": ""})).await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Sending an empty filter is forbidden.",
|
|
"code": "invalid_document_filter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_filter"
|
|
}
|
|
"###);
|
|
|
|
// do not send any filter
|
|
let (response, code) = index.delete_document_by_filter(json!({})).await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Missing field `filter`",
|
|
"code": "missing_document_filter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#missing_document_filter"
|
|
}
|
|
"###);
|
|
|
|
// index does not exists
|
|
let (response, code) =
|
|
index.delete_document_by_filter(json!({ "filter": "doggo = bernese"})).await;
|
|
snapshot!(code, @"202 Accepted");
|
|
let response = server.wait_task(response["taskUid"].as_u64().unwrap()).await;
|
|
snapshot!(json_string!(response, { ".duration" => "[duration]", ".enqueuedAt" => "[date]", ".startedAt" => "[date]", ".finishedAt" => "[date]"}), @r###"
|
|
{
|
|
"uid": 0,
|
|
"indexUid": "doggo",
|
|
"status": "failed",
|
|
"type": "documentDeletion",
|
|
"canceledBy": null,
|
|
"details": {
|
|
"providedIds": 0,
|
|
"deletedDocuments": 0,
|
|
"originalFilter": "\"doggo = bernese\""
|
|
},
|
|
"error": {
|
|
"message": "Index `doggo` not found.",
|
|
"code": "index_not_found",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#index_not_found"
|
|
},
|
|
"duration": "[duration]",
|
|
"enqueuedAt": "[date]",
|
|
"startedAt": "[date]",
|
|
"finishedAt": "[date]"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.create(None).await;
|
|
snapshot!(code, @"202 Accepted");
|
|
server.wait_task(response["taskUid"].as_u64().unwrap()).await;
|
|
|
|
// no filterable are set
|
|
let (response, code) =
|
|
index.delete_document_by_filter(json!({ "filter": "doggo = bernese"})).await;
|
|
snapshot!(code, @"202 Accepted");
|
|
let response = server.wait_task(response["taskUid"].as_u64().unwrap()).await;
|
|
snapshot!(json_string!(response, { ".duration" => "[duration]", ".enqueuedAt" => "[date]", ".startedAt" => "[date]", ".finishedAt" => "[date]"}), @r###"
|
|
{
|
|
"uid": 2,
|
|
"indexUid": "doggo",
|
|
"status": "failed",
|
|
"type": "documentDeletion",
|
|
"canceledBy": null,
|
|
"details": {
|
|
"providedIds": 0,
|
|
"deletedDocuments": 0,
|
|
"originalFilter": "\"doggo = bernese\""
|
|
},
|
|
"error": {
|
|
"message": "Attribute `doggo` is not filterable. This index does not have configured filterable attributes.\n1:6 doggo = bernese",
|
|
"code": "invalid_document_filter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_filter"
|
|
},
|
|
"duration": "[duration]",
|
|
"enqueuedAt": "[date]",
|
|
"startedAt": "[date]",
|
|
"finishedAt": "[date]"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.update_settings_filterable_attributes(json!(["doggo"])).await;
|
|
snapshot!(code, @"202 Accepted");
|
|
server.wait_task(response["taskUid"].as_u64().unwrap()).await;
|
|
|
|
// not filterable while there is a filterable attribute
|
|
let (response, code) =
|
|
index.delete_document_by_filter(json!({ "filter": "catto = jorts"})).await;
|
|
snapshot!(code, @"202 Accepted");
|
|
let response = server.wait_task(response["taskUid"].as_u64().unwrap()).await;
|
|
snapshot!(json_string!(response, { ".duration" => "[duration]", ".enqueuedAt" => "[date]", ".startedAt" => "[date]", ".finishedAt" => "[date]"}), @r###"
|
|
{
|
|
"uid": 4,
|
|
"indexUid": "doggo",
|
|
"status": "failed",
|
|
"type": "documentDeletion",
|
|
"canceledBy": null,
|
|
"details": {
|
|
"providedIds": 0,
|
|
"deletedDocuments": 0,
|
|
"originalFilter": "\"catto = jorts\""
|
|
},
|
|
"error": {
|
|
"message": "Attribute `catto` is not filterable. Available filterable attributes are: `doggo`.\n1:6 catto = jorts",
|
|
"code": "invalid_document_filter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_filter"
|
|
},
|
|
"duration": "[duration]",
|
|
"enqueuedAt": "[date]",
|
|
"startedAt": "[date]",
|
|
"finishedAt": "[date]"
|
|
}
|
|
"###);
|
|
}
|
|
|
|
#[actix_rt::test]
|
|
async fn fetch_document_by_filter() {
|
|
let server = Server::new().await;
|
|
let index = server.index("doggo");
|
|
index.update_settings_filterable_attributes(json!(["color"])).await;
|
|
index
|
|
.add_documents(
|
|
json!([
|
|
{ "id": 0, "color": "red" },
|
|
{ "id": 1, "color": "blue" },
|
|
{ "id": 2, "color": "blue" },
|
|
{ "id": 3 },
|
|
]),
|
|
Some("id"),
|
|
)
|
|
.await;
|
|
index.wait_task(1).await;
|
|
|
|
let (response, code) = index.get_document_by_filter(json!(null)).await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid value type: expected an object, but found null",
|
|
"code": "bad_request",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#bad_request"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.get_document_by_filter(json!({ "offset": "doggo" })).await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid value type at `.offset`: expected a positive integer, but found a string: `\"doggo\"`",
|
|
"code": "invalid_document_offset",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_offset"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.get_document_by_filter(json!({ "limit": "doggo" })).await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid value type at `.limit`: expected a positive integer, but found a string: `\"doggo\"`",
|
|
"code": "invalid_document_limit",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_limit"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.get_document_by_filter(json!({ "fields": "doggo" })).await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid value type at `.fields`: expected an array, but found a string: `\"doggo\"`",
|
|
"code": "invalid_document_fields",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_fields"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.get_document_by_filter(json!({ "filter": true })).await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Invalid syntax for the filter parameter: `expected String, Array, found: true`.",
|
|
"code": "invalid_document_filter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_filter"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) = index.get_document_by_filter(json!({ "filter": "cool doggo" })).await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` at `cool doggo`.\n1:11 cool doggo",
|
|
"code": "invalid_document_filter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_filter"
|
|
}
|
|
"###);
|
|
|
|
let (response, code) =
|
|
index.get_document_by_filter(json!({ "filter": "doggo = bernese" })).await;
|
|
snapshot!(code, @"400 Bad Request");
|
|
snapshot!(json_string!(response), @r###"
|
|
{
|
|
"message": "Attribute `doggo` is not filterable. Available filterable attributes are: `color`.\n1:6 doggo = bernese",
|
|
"code": "invalid_document_filter",
|
|
"type": "invalid_request",
|
|
"link": "https://docs.meilisearch.com/errors#invalid_document_filter"
|
|
}
|
|
"###);
|
|
}
|