mirror of
				https://github.com/meilisearch/meilisearch.git
				synced 2025-10-25 21:16:28 +00:00 
			
		
		
		
	Refactor query parameter deserialisation logic
This commit is contained in:
		
							
								
								
									
										31
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										31
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							| @@ -1026,7 +1026,18 @@ dependencies = [ | ||||
| name = "deserr" | ||||
| version = "0.1.4" | ||||
| dependencies = [ | ||||
|  "deserr-internal", | ||||
|  "deserr-internal 0.1.4", | ||||
|  "serde-cs", | ||||
|  "serde_json", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "deserr" | ||||
| version = "0.1.4" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "86290491a2b5c21a1a5083da8dae831006761258fabd5617309c3eebc5f89468" | ||||
| dependencies = [ | ||||
|  "deserr-internal 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", | ||||
|  "serde-cs", | ||||
|  "serde_json", | ||||
| ] | ||||
| @@ -1041,6 +1052,18 @@ dependencies = [ | ||||
|  "syn 1.0.107", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "deserr-internal" | ||||
| version = "0.1.4" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "7131de1c27581bc376a22166c9f570be91b76cb096be2f6aecf224c27bf7c49a" | ||||
| dependencies = [ | ||||
|  "convert_case 0.5.0", | ||||
|  "proc-macro2", | ||||
|  "quote", | ||||
|  "syn", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "deunicode" | ||||
| version = "1.3.3" | ||||
| @@ -2300,7 +2323,7 @@ dependencies = [ | ||||
|  "cargo_toml", | ||||
|  "clap 4.0.32", | ||||
|  "crossbeam-channel", | ||||
|  "deserr", | ||||
|  "deserr 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", | ||||
|  "dump", | ||||
|  "either", | ||||
|  "env_logger", | ||||
| @@ -2391,7 +2414,7 @@ dependencies = [ | ||||
|  "anyhow", | ||||
|  "convert_case 0.6.0", | ||||
|  "csv", | ||||
|  "deserr", | ||||
|  "deserr 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", | ||||
|  "either", | ||||
|  "enum-iterator", | ||||
|  "file-store", | ||||
| @@ -2451,7 +2474,7 @@ dependencies = [ | ||||
|  "concat-arrays", | ||||
|  "crossbeam-channel", | ||||
|  "csv", | ||||
|  "deserr", | ||||
|  "deserr 0.1.4", | ||||
|  "either", | ||||
|  "filter-parser", | ||||
|  "flatten-serde-json", | ||||
|   | ||||
| @@ -3,7 +3,6 @@ pub mod error; | ||||
| mod store; | ||||
|  | ||||
| use std::collections::{HashMap, HashSet}; | ||||
| use std::ops::Deref; | ||||
| use std::path::Path; | ||||
| use std::sync::Arc; | ||||
|  | ||||
| @@ -86,15 +85,13 @@ impl AuthController { | ||||
|                     key.indexes | ||||
|                         .into_iter() | ||||
|                         .filter_map(|index| { | ||||
|                             search_rules.get_index_search_rules(index.deref()).map( | ||||
|                                 |index_search_rules| { | ||||
|                                     (String::from(index), Some(index_search_rules)) | ||||
|                                 }, | ||||
|                             search_rules.get_index_search_rules(&format!("{index}")).map( | ||||
|                                 |index_search_rules| (index.to_string(), Some(index_search_rules)), | ||||
|                             ) | ||||
|                         }) | ||||
|                         .collect(), | ||||
|                 ), | ||||
|                 None => SearchRules::Set(key.indexes.into_iter().map(String::from).collect()), | ||||
|                 None => SearchRules::Set(key.indexes.into_iter().map(|x| x.to_string()).collect()), | ||||
|             }; | ||||
|         } else if let Some(search_rules) = search_rules { | ||||
|             filters.search_rules = search_rules; | ||||
|   | ||||
| @@ -3,7 +3,6 @@ use std::cmp::Reverse; | ||||
| use std::collections::HashSet; | ||||
| use std::convert::{TryFrom, TryInto}; | ||||
| use std::fs::create_dir_all; | ||||
| use std::ops::Deref; | ||||
| use std::path::Path; | ||||
| use std::str; | ||||
| use std::sync::Arc; | ||||
| @@ -135,7 +134,7 @@ impl HeedAuthStore { | ||||
|                 for index in key.indexes.iter() { | ||||
|                     db.put( | ||||
|                         &mut wtxn, | ||||
|                         &(&uid, &action, Some(index.deref().as_bytes())), | ||||
|                         &(&uid, &action, Some(index.to_string().as_bytes())), | ||||
|                         &key.expires_at, | ||||
|                     )?; | ||||
|                 } | ||||
|   | ||||
| @@ -9,7 +9,7 @@ actix-web = { version = "4.2.1", default-features = false } | ||||
| anyhow = "1.0.65" | ||||
| convert_case = "0.6.0" | ||||
| csv = "1.1.6" | ||||
| deserr = { path = "/Users/meilisearch/Documents/deserr" } | ||||
| deserr = "0.1.4" | ||||
| either = { version = "1.6.1", features = ["serde"] } | ||||
| enum-iterator = "1.1.3" | ||||
| file-store = { path = "../file-store" } | ||||
|   | ||||
							
								
								
									
										315
									
								
								meilisearch-types/src/deserr/error_messages.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										315
									
								
								meilisearch-types/src/deserr/error_messages.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,315 @@ | ||||
| /*! | ||||
| This module implements the error messages of deserialization errors. | ||||
|  | ||||
| We try to: | ||||
| 1. Give a human-readable description of where the error originated. | ||||
| 2. Use the correct terms depending on the format of the request (json/query param) | ||||
| 3. Categorise the type of the error (e.g. missing field, wrong value type, unexpected error, etc.) | ||||
|  */ | ||||
| use deserr::{ErrorKind, IntoValue, ValueKind, ValuePointerRef}; | ||||
|  | ||||
| use super::{DeserrJsonError, DeserrQueryParamError}; | ||||
| use crate::error::ErrorCode; | ||||
|  | ||||
| /// Return a description of the given location in a Json, preceded by the given article. | ||||
| /// e.g. `at .key1[8].key2`. If the location is the origin, the given article will not be | ||||
| /// included in the description. | ||||
| pub fn location_json_description(location: ValuePointerRef, article: &str) -> String { | ||||
|     fn rec(location: ValuePointerRef) -> String { | ||||
|         match location { | ||||
|             ValuePointerRef::Origin => String::new(), | ||||
|             ValuePointerRef::Key { key, prev } => rec(*prev) + "." + key, | ||||
|             ValuePointerRef::Index { index, prev } => format!("{}[{index}]", rec(*prev)), | ||||
|         } | ||||
|     } | ||||
|     match location { | ||||
|         ValuePointerRef::Origin => String::new(), | ||||
|         _ => { | ||||
|             format!("{article} `{}`", rec(location)) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Return a description of the list of value kinds for a Json payload. | ||||
| fn value_kinds_description_json(kinds: &[ValueKind]) -> String { | ||||
|     // Rank each value kind so that they can be sorted (and deduplicated) | ||||
|     // Having a predictable order helps with pattern matching | ||||
|     fn order(kind: &ValueKind) -> u8 { | ||||
|         match kind { | ||||
|             ValueKind::Null => 0, | ||||
|             ValueKind::Boolean => 1, | ||||
|             ValueKind::Integer => 2, | ||||
|             ValueKind::NegativeInteger => 3, | ||||
|             ValueKind::Float => 4, | ||||
|             ValueKind::String => 5, | ||||
|             ValueKind::Sequence => 6, | ||||
|             ValueKind::Map => 7, | ||||
|         } | ||||
|     } | ||||
|     // Return a description of a single value kind, preceded by an article | ||||
|     fn single_description(kind: &ValueKind) -> &'static str { | ||||
|         match kind { | ||||
|             ValueKind::Null => "null", | ||||
|             ValueKind::Boolean => "a boolean", | ||||
|             ValueKind::Integer => "a positive integer", | ||||
|             ValueKind::NegativeInteger => "an integer", | ||||
|             ValueKind::Float => "a number", | ||||
|             ValueKind::String => "a string", | ||||
|             ValueKind::Sequence => "an array", | ||||
|             ValueKind::Map => "an object", | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn description_rec(kinds: &[ValueKind], count_items: &mut usize, message: &mut String) { | ||||
|         let (msg_part, rest): (_, &[ValueKind]) = match kinds { | ||||
|             [] => (String::new(), &[]), | ||||
|             [ValueKind::Integer | ValueKind::NegativeInteger, ValueKind::Float, rest @ ..] => { | ||||
|                 ("a number".to_owned(), rest) | ||||
|             } | ||||
|             [ValueKind::Integer, ValueKind::NegativeInteger, ValueKind::Float, rest @ ..] => { | ||||
|                 ("a number".to_owned(), rest) | ||||
|             } | ||||
|             [ValueKind::Integer, ValueKind::NegativeInteger, rest @ ..] => { | ||||
|                 ("an integer".to_owned(), rest) | ||||
|             } | ||||
|             [a] => (single_description(a).to_owned(), &[]), | ||||
|             [a, rest @ ..] => (single_description(a).to_owned(), rest), | ||||
|         }; | ||||
|  | ||||
|         if rest.is_empty() { | ||||
|             if *count_items == 0 { | ||||
|                 message.push_str(&msg_part); | ||||
|             } else if *count_items == 1 { | ||||
|                 message.push_str(&format!(" or {msg_part}")); | ||||
|             } else { | ||||
|                 message.push_str(&format!(", or {msg_part}")); | ||||
|             } | ||||
|         } else { | ||||
|             if *count_items == 0 { | ||||
|                 message.push_str(&msg_part); | ||||
|             } else { | ||||
|                 message.push_str(&format!(", {msg_part}")); | ||||
|             } | ||||
|  | ||||
|             *count_items += 1; | ||||
|             description_rec(rest, count_items, message); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     let mut kinds = kinds.to_owned(); | ||||
|     kinds.sort_by_key(order); | ||||
|     kinds.dedup(); | ||||
|  | ||||
|     if kinds.is_empty() { | ||||
|         // Should not happen ideally | ||||
|         "a different value".to_owned() | ||||
|     } else { | ||||
|         let mut message = String::new(); | ||||
|         description_rec(kinds.as_slice(), &mut 0, &mut message); | ||||
|         message | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Return the JSON string of the value preceded by a description of its kind | ||||
| fn value_description_with_kind_json(v: &serde_json::Value) -> String { | ||||
|     match v.kind() { | ||||
|         ValueKind::Null => "null".to_owned(), | ||||
|         kind => { | ||||
|             format!( | ||||
|                 "{}: `{}`", | ||||
|                 value_kinds_description_json(&[kind]), | ||||
|                 serde_json::to_string(v).unwrap() | ||||
|             ) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<C: Default + ErrorCode> deserr::DeserializeError for DeserrJsonError<C> { | ||||
|     fn error<V: IntoValue>( | ||||
|         _self_: Option<Self>, | ||||
|         error: deserr::ErrorKind<V>, | ||||
|         location: ValuePointerRef, | ||||
|     ) -> Result<Self, Self> { | ||||
|         let mut message = String::new(); | ||||
|  | ||||
|         message.push_str(&match error { | ||||
|             ErrorKind::IncorrectValueKind { actual, accepted } => { | ||||
|                 let expected = value_kinds_description_json(accepted); | ||||
|                 let received = value_description_with_kind_json(&serde_json::Value::from(actual)); | ||||
|  | ||||
|                 let location = location_json_description(location, " at"); | ||||
|  | ||||
|                 format!("Invalid value type{location}: expected {expected}, but found {received}") | ||||
|             } | ||||
|             ErrorKind::MissingField { field } => { | ||||
|                 let location = location_json_description(location, " inside"); | ||||
|                 format!("Missing field `{field}`{location}") | ||||
|             } | ||||
|             ErrorKind::UnknownKey { key, accepted } => { | ||||
|                 let location = location_json_description(location, " inside"); | ||||
|                 format!( | ||||
|                     "Unknown field `{}`{location}: expected one of {}", | ||||
|                     key, | ||||
|                     accepted | ||||
|                         .iter() | ||||
|                         .map(|accepted| format!("`{}`", accepted)) | ||||
|                         .collect::<Vec<String>>() | ||||
|                         .join(", ") | ||||
|                 ) | ||||
|             } | ||||
|             ErrorKind::UnknownValue { value, accepted } => { | ||||
|                 let location = location_json_description(location, " at"); | ||||
|                 format!( | ||||
|                     "Unknown value `{}`{location}: expected one of {}", | ||||
|                     value, | ||||
|                     accepted | ||||
|                         .iter() | ||||
|                         .map(|accepted| format!("`{}`", accepted)) | ||||
|                         .collect::<Vec<String>>() | ||||
|                         .join(", "), | ||||
|                 ) | ||||
|             } | ||||
|             ErrorKind::Unexpected { msg } => { | ||||
|                 let location = location_json_description(location, " at"); | ||||
|                 format!("Invalid value{location}: {msg}") | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         Err(DeserrJsonError::new(message, C::default().error_code())) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Return a description of the given location in query parameters, preceded by the | ||||
| /// given article. e.g. `at key5[2]`. If the location is the origin, the given article | ||||
| /// will not be included in the description. | ||||
| pub fn location_query_param_description(location: ValuePointerRef, article: &str) -> String { | ||||
|     fn rec(location: ValuePointerRef) -> String { | ||||
|         match location { | ||||
|             ValuePointerRef::Origin => String::new(), | ||||
|             ValuePointerRef::Key { key, prev } => { | ||||
|                 if matches!(prev, ValuePointerRef::Origin) { | ||||
|                     key.to_owned() | ||||
|                 } else { | ||||
|                     rec(*prev) + "." + key | ||||
|                 } | ||||
|             } | ||||
|             ValuePointerRef::Index { index, prev } => format!("{}[{index}]", rec(*prev)), | ||||
|         } | ||||
|     } | ||||
|     match location { | ||||
|         ValuePointerRef::Origin => String::new(), | ||||
|         _ => { | ||||
|             format!("{article} `{}`", rec(location)) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<C: Default + ErrorCode> deserr::DeserializeError for DeserrQueryParamError<C> { | ||||
|     fn error<V: IntoValue>( | ||||
|         _self_: Option<Self>, | ||||
|         error: deserr::ErrorKind<V>, | ||||
|         location: ValuePointerRef, | ||||
|     ) -> Result<Self, Self> { | ||||
|         let mut message = String::new(); | ||||
|  | ||||
|         message.push_str(&match error { | ||||
|             ErrorKind::IncorrectValueKind { actual, accepted } => { | ||||
|                 let expected = value_kinds_description_query_param(accepted); | ||||
|                 let received = value_description_with_kind_query_param(actual); | ||||
|  | ||||
|                 let location = location_query_param_description(location, " for parameter"); | ||||
|  | ||||
|                 format!("Invalid value type{location}: expected {expected}, but found {received}") | ||||
|             } | ||||
|             ErrorKind::MissingField { field } => { | ||||
|                 let location = location_query_param_description(location, " inside"); | ||||
|                 format!("Missing parameter `{field}`{location}") | ||||
|             } | ||||
|             ErrorKind::UnknownKey { key, accepted } => { | ||||
|                 let location = location_query_param_description(location, " inside"); | ||||
|                 format!( | ||||
|                     "Unknown parameter `{}`{location}: expected one of {}", | ||||
|                     key, | ||||
|                     accepted | ||||
|                         .iter() | ||||
|                         .map(|accepted| format!("`{}`", accepted)) | ||||
|                         .collect::<Vec<String>>() | ||||
|                         .join(", ") | ||||
|                 ) | ||||
|             } | ||||
|             ErrorKind::UnknownValue { value, accepted } => { | ||||
|                 let location = location_query_param_description(location, " for parameter"); | ||||
|                 format!( | ||||
|                     "Unknown value `{}`{location}: expected one of {}", | ||||
|                     value, | ||||
|                     accepted | ||||
|                         .iter() | ||||
|                         .map(|accepted| format!("`{}`", accepted)) | ||||
|                         .collect::<Vec<String>>() | ||||
|                         .join(", "), | ||||
|                 ) | ||||
|             } | ||||
|             ErrorKind::Unexpected { msg } => { | ||||
|                 let location = location_query_param_description(location, " in parameter"); | ||||
|                 format!("Invalid value{location}: {msg}") | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         Err(DeserrQueryParamError::new(message, C::default().error_code())) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Return a description of the list of value kinds for query parameters | ||||
| /// Since query parameters are always treated as strings, we always return | ||||
| /// "a string" for now. | ||||
| fn value_kinds_description_query_param(_accepted: &[ValueKind]) -> String { | ||||
|     "a string".to_owned() | ||||
| } | ||||
|  | ||||
| fn value_description_with_kind_query_param<V: IntoValue>(actual: deserr::Value<V>) -> String { | ||||
|     match actual { | ||||
|         deserr::Value::Null => "null".to_owned(), | ||||
|         deserr::Value::Boolean(x) => format!("a boolean: `{x}`"), | ||||
|         deserr::Value::Integer(x) => format!("an integer: `{x}`"), | ||||
|         deserr::Value::NegativeInteger(x) => { | ||||
|             format!("an integer: `{x}`") | ||||
|         } | ||||
|         deserr::Value::Float(x) => { | ||||
|             format!("a number: `{x}`") | ||||
|         } | ||||
|         deserr::Value::String(x) => { | ||||
|             format!("a string: `{x}`") | ||||
|         } | ||||
|         deserr::Value::Sequence(_) => "multiple values".to_owned(), | ||||
|         deserr::Value::Map(_) => "multiple parameters".to_owned(), | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg(test)] | ||||
| mod tests { | ||||
|     use deserr::ValueKind; | ||||
|  | ||||
|     use crate::deserr::error_messages::value_kinds_description_json; | ||||
|  | ||||
|     #[test] | ||||
|     fn test_value_kinds_description_json() { | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[]), @"a different value"); | ||||
|  | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Boolean]), @"a boolean"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Integer]), @"a positive integer"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::NegativeInteger]), @"an integer"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Integer]), @"a positive integer"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::String]), @"a string"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Sequence]), @"an array"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Map]), @"an object"); | ||||
|  | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Integer, ValueKind::Boolean]), @"a boolean or a positive integer"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Null, ValueKind::Integer]), @"null or a positive integer"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Sequence, ValueKind::NegativeInteger]), @"an integer or an array"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Integer, ValueKind::Float]), @"a number"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Integer, ValueKind::Float, ValueKind::NegativeInteger]), @"a number"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Integer, ValueKind::Float, ValueKind::NegativeInteger, ValueKind::Null]), @"null or a number"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Boolean, ValueKind::Integer, ValueKind::Float, ValueKind::NegativeInteger, ValueKind::Null]), @"null, a boolean, or a number"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Null, ValueKind::Boolean, ValueKind::Integer, ValueKind::Float, ValueKind::NegativeInteger, ValueKind::Null]), @"null, a boolean, or a number"); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										134
									
								
								meilisearch-types/src/deserr/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										134
									
								
								meilisearch-types/src/deserr/mod.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,134 @@ | ||||
| use std::convert::Infallible; | ||||
| use std::fmt; | ||||
| use std::marker::PhantomData; | ||||
|  | ||||
| use deserr::{DeserializeError, MergeWithError, ValuePointerRef}; | ||||
|  | ||||
| use crate::error::deserr_codes::{self, *}; | ||||
| use crate::error::{ | ||||
|     unwrap_any, Code, DeserrParseBoolError, DeserrParseIntError, ErrorCode, InvalidTaskDateError, | ||||
|     ParseOffsetDateTimeError, | ||||
| }; | ||||
| use crate::index_uid::IndexUidFormatError; | ||||
| use crate::tasks::{ParseTaskKindError, ParseTaskStatusError}; | ||||
|  | ||||
| pub mod error_messages; | ||||
| pub mod query_params; | ||||
|  | ||||
| /// Marker type for the Json format | ||||
| pub struct DeserrJson; | ||||
| /// Marker type for the Query Parameter format | ||||
| pub struct DeserrQueryParam; | ||||
|  | ||||
| pub type DeserrJsonError<C = deserr_codes::BadRequest> = DeserrError<DeserrJson, C>; | ||||
| pub type DeserrQueryParamError<C = deserr_codes::BadRequest> = DeserrError<DeserrQueryParam, C>; | ||||
|  | ||||
| /// A request deserialization error. | ||||
| /// | ||||
| /// The first generic paramater is a marker type describing the format of the request: either json (e.g. [`DeserrJson`] or [`DeserrQueryParam`]). | ||||
| /// The second generic parameter is the default error code for the deserialization error, in case it is not given. | ||||
| pub struct DeserrError<Format, C: Default + ErrorCode> { | ||||
|     pub msg: String, | ||||
|     pub code: Code, | ||||
|     _phantom: PhantomData<(Format, C)>, | ||||
| } | ||||
| impl<Format, C: Default + ErrorCode> DeserrError<Format, C> { | ||||
|     pub fn new(msg: String, code: Code) -> Self { | ||||
|         Self { msg, code, _phantom: PhantomData } | ||||
|     } | ||||
| } | ||||
| impl<Format, C: Default + ErrorCode> std::fmt::Debug for DeserrError<Format, C> { | ||||
|     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||||
|         f.debug_struct("DeserrError").field("msg", &self.msg).field("code", &self.code).finish() | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<Format, C: Default + ErrorCode> std::fmt::Display for DeserrError<Format, C> { | ||||
|     fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||||
|         write!(f, "{}", self.msg) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<Format, C: Default + ErrorCode> std::error::Error for DeserrError<Format, C> {} | ||||
| impl<Format, C: Default + ErrorCode> ErrorCode for DeserrError<Format, C> { | ||||
|     fn error_code(&self) -> Code { | ||||
|         self.code | ||||
|     } | ||||
| } | ||||
|  | ||||
| // For now, we don't accumulate errors. Only one deserialisation error is ever returned at a time. | ||||
| impl<Format, C1: Default + ErrorCode, C2: Default + ErrorCode> | ||||
|     MergeWithError<DeserrError<Format, C2>> for DeserrError<Format, C1> | ||||
| { | ||||
|     fn merge( | ||||
|         _self_: Option<Self>, | ||||
|         other: DeserrError<Format, C2>, | ||||
|         _merge_location: ValuePointerRef, | ||||
|     ) -> Result<Self, Self> { | ||||
|         Err(DeserrError { msg: other.msg, code: other.code, _phantom: PhantomData }) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<Format, C: Default + ErrorCode> MergeWithError<Infallible> for DeserrError<Format, C> { | ||||
|     fn merge( | ||||
|         _self_: Option<Self>, | ||||
|         _other: Infallible, | ||||
|         _merge_location: ValuePointerRef, | ||||
|     ) -> Result<Self, Self> { | ||||
|         unreachable!() | ||||
|     } | ||||
| } | ||||
|  | ||||
| // Implement a convenience function to build a `missing_field` error | ||||
| macro_rules! make_missing_field_convenience_builder { | ||||
|     ($err_code:ident, $fn_name:ident) => { | ||||
|         impl DeserrJsonError<$err_code> { | ||||
|             pub fn $fn_name(field: &str, location: ValuePointerRef) -> Self { | ||||
|                 let x = unwrap_any(Self::error::<Infallible>( | ||||
|                     None, | ||||
|                     deserr::ErrorKind::MissingField { field }, | ||||
|                     location, | ||||
|                 )); | ||||
|                 Self { msg: x.msg, code: $err_code.error_code(), _phantom: PhantomData } | ||||
|             } | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| make_missing_field_convenience_builder!(MissingIndexUid, missing_index_uid); | ||||
| make_missing_field_convenience_builder!(MissingApiKeyActions, missing_api_key_actions); | ||||
| make_missing_field_convenience_builder!(MissingApiKeyExpiresAt, missing_api_key_expires_at); | ||||
| make_missing_field_convenience_builder!(MissingApiKeyIndexes, missing_api_key_indexes); | ||||
| make_missing_field_convenience_builder!(MissingSwapIndexes, missing_swap_indexes); | ||||
|  | ||||
| // Integrate a sub-error into a [`DeserrError`] by taking its error message but using | ||||
| // the default error code (C) from `Self` | ||||
| macro_rules! merge_with_error_impl_take_error_message { | ||||
|     ($err_type:ty) => { | ||||
|         impl<Format, C: Default + ErrorCode> MergeWithError<$err_type> for DeserrError<Format, C> | ||||
|         where | ||||
|             DeserrError<Format, C>: deserr::DeserializeError, | ||||
|         { | ||||
|             fn merge( | ||||
|                 _self_: Option<Self>, | ||||
|                 other: $err_type, | ||||
|                 merge_location: ValuePointerRef, | ||||
|             ) -> Result<Self, Self> { | ||||
|                 DeserrError::<Format, C>::error::<Infallible>( | ||||
|                     None, | ||||
|                     deserr::ErrorKind::Unexpected { msg: other.to_string() }, | ||||
|                     merge_location, | ||||
|                 ) | ||||
|             } | ||||
|         } | ||||
|     }; | ||||
| } | ||||
|  | ||||
| // All these errors can be merged into a `DeserrError` | ||||
| merge_with_error_impl_take_error_message!(DeserrParseIntError); | ||||
| merge_with_error_impl_take_error_message!(DeserrParseBoolError); | ||||
| merge_with_error_impl_take_error_message!(uuid::Error); | ||||
| merge_with_error_impl_take_error_message!(InvalidTaskDateError); | ||||
| merge_with_error_impl_take_error_message!(ParseOffsetDateTimeError); | ||||
| merge_with_error_impl_take_error_message!(ParseTaskKindError); | ||||
| merge_with_error_impl_take_error_message!(ParseTaskStatusError); | ||||
| merge_with_error_impl_take_error_message!(IndexUidFormatError); | ||||
							
								
								
									
										115
									
								
								meilisearch-types/src/deserr/query_params.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										115
									
								
								meilisearch-types/src/deserr/query_params.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,115 @@ | ||||
| /*! | ||||
| This module provides helper traits, types, and functions to deserialize query parameters. | ||||
|  | ||||
| The source of the problem is that query parameters only give us a string to work with. | ||||
| This means `deserr` is never given a sequence or numbers, and thus the default deserialization | ||||
| code for common types such as `usize` or `Vec<T>` does not work. To work around it, we create a | ||||
| wrapper type called `Param<T>`, which is deserialised using the `from_query_param` method of the trait | ||||
| `FromQueryParameter`. | ||||
|  | ||||
| We also use other helper types such as `CS` (i.e. comma-separated) from `serde_cs` as well as | ||||
| `StarOr`, `OptionStarOr`, and `OptionStarOrList`. | ||||
| */ | ||||
|  | ||||
| use std::convert::Infallible; | ||||
| use std::ops::Deref; | ||||
| use std::str::FromStr; | ||||
|  | ||||
| use deserr::{DeserializeError, DeserializeFromValue, MergeWithError, ValueKind}; | ||||
|  | ||||
| use super::{DeserrParseBoolError, DeserrParseIntError}; | ||||
| use crate::error::unwrap_any; | ||||
| use crate::index_uid::IndexUid; | ||||
| use crate::tasks::{Kind, Status}; | ||||
|  | ||||
| /// A wrapper type indicating that the inner value should be | ||||
| /// deserialised from a query parameter string. | ||||
| /// | ||||
| /// Note that if the field is optional, it is better to use | ||||
| /// `Option<Param<T>>` instead of `Param<Option<T>>`. | ||||
| #[derive(Default, Debug, Clone, Copy)] | ||||
| pub struct Param<T>(pub T); | ||||
|  | ||||
| impl<T> Deref for Param<T> { | ||||
|     type Target = T; | ||||
|  | ||||
|     fn deref(&self) -> &Self::Target { | ||||
|         &self.0 | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<T, E> DeserializeFromValue<E> for Param<T> | ||||
| where | ||||
|     E: DeserializeError + MergeWithError<T::Err>, | ||||
|     T: FromQueryParameter, | ||||
| { | ||||
|     fn deserialize_from_value<V: deserr::IntoValue>( | ||||
|         value: deserr::Value<V>, | ||||
|         location: deserr::ValuePointerRef, | ||||
|     ) -> Result<Self, E> { | ||||
|         match value { | ||||
|             deserr::Value::String(s) => match T::from_query_param(&s) { | ||||
|                 Ok(x) => Ok(Param(x)), | ||||
|                 Err(e) => Err(unwrap_any(E::merge(None, e, location))), | ||||
|             }, | ||||
|             _ => Err(unwrap_any(E::error( | ||||
|                 None, | ||||
|                 deserr::ErrorKind::IncorrectValueKind { | ||||
|                     actual: value, | ||||
|                     accepted: &[ValueKind::String], | ||||
|                 }, | ||||
|                 location, | ||||
|             ))), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Parse a value from a query parameter string. | ||||
| /// | ||||
| /// This trait is functionally equivalent to `FromStr`. | ||||
| /// Having a separate trait trait allows us to return better | ||||
| /// deserializatio error messages. | ||||
| pub trait FromQueryParameter: Sized { | ||||
|     type Err; | ||||
|     fn from_query_param(p: &str) -> Result<Self, Self::Err>; | ||||
| } | ||||
|  | ||||
| /// Implement `FromQueryParameter` for the given type using its `FromStr` | ||||
| /// trait implementation. | ||||
| macro_rules! impl_from_query_param_from_str { | ||||
|     ($type:ty) => { | ||||
|         impl FromQueryParameter for $type { | ||||
|             type Err = <$type as FromStr>::Err; | ||||
|             fn from_query_param(p: &str) -> Result<Self, Self::Err> { | ||||
|                 p.parse() | ||||
|             } | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| impl_from_query_param_from_str!(Kind); | ||||
| impl_from_query_param_from_str!(Status); | ||||
| impl_from_query_param_from_str!(IndexUid); | ||||
|  | ||||
| /// Implement `FromQueryParameter` for the given type using its `FromStr` | ||||
| /// trait implementation, replacing the returned error with a struct | ||||
| /// that wraps the original query parameter. | ||||
| macro_rules! impl_from_query_param_wrap_original_value_in_error { | ||||
|     ($type:ty, $err_type:path) => { | ||||
|         impl FromQueryParameter for $type { | ||||
|             type Err = $err_type; | ||||
|             fn from_query_param(p: &str) -> Result<Self, Self::Err> { | ||||
|                 p.parse().map_err(|_| $err_type(p.to_owned())) | ||||
|             } | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| impl_from_query_param_wrap_original_value_in_error!(usize, DeserrParseIntError); | ||||
| impl_from_query_param_wrap_original_value_in_error!(u32, DeserrParseIntError); | ||||
| impl_from_query_param_wrap_original_value_in_error!(bool, DeserrParseBoolError); | ||||
|  | ||||
| impl FromQueryParameter for String { | ||||
|     type Err = Infallible; | ||||
|     fn from_query_param(p: &str) -> Result<Self, Infallible> { | ||||
|         Ok(p.to_owned()) | ||||
|     } | ||||
| } | ||||
| @@ -1,30 +1,17 @@ | ||||
| use std::convert::Infallible; | ||||
| use std::marker::PhantomData; | ||||
| use std::str::FromStr; | ||||
| use std::{fmt, io}; | ||||
|  | ||||
| use actix_web::http::StatusCode; | ||||
| use actix_web::{self as aweb, HttpResponseBuilder}; | ||||
| use aweb::rt::task::JoinError; | ||||
| use convert_case::Casing; | ||||
| use deserr::{DeserializeError, ErrorKind, IntoValue, MergeWithError, ValueKind, ValuePointerRef}; | ||||
| use milli::heed::{Error as HeedError, MdbError}; | ||||
| use serde::{Deserialize, Serialize}; | ||||
| use serde_cs::vec::CS; | ||||
|  | ||||
| use crate::star_or::StarOr; | ||||
|  | ||||
| use self::deserr_codes::{ | ||||
|     InvalidSwapIndexes, MissingApiKeyActions, MissingApiKeyExpiresAt, MissingApiKeyIndexes, | ||||
|     MissingIndexUid, MissingSwapIndexes, | ||||
| }; | ||||
|  | ||||
| #[derive(Debug, Serialize, Deserialize, Clone, PartialEq, Eq)] | ||||
| #[serde(rename_all = "camelCase")] | ||||
| #[cfg_attr(feature = "test-traits", derive(proptest_derive::Arbitrary))] | ||||
| pub struct ResponseError { | ||||
|     #[serde(skip)] | ||||
|     #[cfg_attr(feature = "test-traits", proptest(strategy = "strategy::status_code_strategy()"))] | ||||
|     code: StatusCode, | ||||
|     message: String, | ||||
|     #[serde(rename = "code")] | ||||
| @@ -43,7 +30,7 @@ impl ResponseError { | ||||
|         Self { | ||||
|             code: code.http(), | ||||
|             message, | ||||
|             error_code: code.err_code().error_name, | ||||
|             error_code: code.name(), | ||||
|             error_type: code.type_(), | ||||
|             error_link: code.url(), | ||||
|         } | ||||
| @@ -104,9 +91,9 @@ pub trait ErrorCode { | ||||
|  | ||||
| #[allow(clippy::enum_variant_names)] | ||||
| enum ErrorType { | ||||
|     InternalError, | ||||
|     InvalidRequestError, | ||||
|     AuthenticationError, | ||||
|     Internal, | ||||
|     InvalidRequest, | ||||
|     Auth, | ||||
|     System, | ||||
| } | ||||
|  | ||||
| @@ -115,14 +102,24 @@ impl fmt::Display for ErrorType { | ||||
|         use ErrorType::*; | ||||
|  | ||||
|         match self { | ||||
|             InternalError => write!(f, "internal"), | ||||
|             InvalidRequestError => write!(f, "invalid_request"), | ||||
|             AuthenticationError => write!(f, "auth"), | ||||
|             Internal => write!(f, "internal"), | ||||
|             InvalidRequest => write!(f, "invalid_request"), | ||||
|             Auth => write!(f, "auth"), | ||||
|             System => write!(f, "system"), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Implement all the error codes. | ||||
| /// | ||||
| /// 1. Make an enum `Code` where each error code is a variant | ||||
| /// 2. Implement the `http`, `name`, and `type_` method on the enum | ||||
| /// 3. Make a unit type for each error code in the module `deserr_codes`. | ||||
| /// | ||||
| /// The unit type's purpose is to be used as a marker type parameter, e.g. | ||||
| /// `DeserrJsonError<MyErrorCode>`. It implements `Default` and `ErrorCode`, | ||||
| /// so we can get a value of the `Code` enum with the correct variant by calling | ||||
| /// `MyErrorCode::default().error_code()`. | ||||
| macro_rules! make_error_codes { | ||||
|     ($($code_ident:ident, $err_type:ident, $status:ident);*) => { | ||||
|         #[derive(Debug, Clone, Copy, PartialEq, Eq)] | ||||
| @@ -130,29 +127,31 @@ macro_rules! make_error_codes { | ||||
|             $($code_ident),* | ||||
|         } | ||||
|         impl Code { | ||||
|             /// associate a `Code` variant to the actual ErrCode | ||||
|             fn err_code(&self) -> ErrCode { | ||||
|                 match self { | ||||
|                     $( | ||||
|                         Code::$code_ident => { | ||||
|                             ErrCode::$err_type( stringify!($code_ident).to_case(convert_case::Case::Snake), StatusCode::$status) | ||||
|                         } | ||||
|                     )* | ||||
|                 } | ||||
|             } | ||||
|             /// return the HTTP status code associated with the `Code` | ||||
|             fn http(&self) -> StatusCode { | ||||
|                 self.err_code().status_code | ||||
|                 match self { | ||||
|                     $( | ||||
|                         Code::$code_ident => StatusCode::$status | ||||
|                     ),* | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             /// return error name, used as error code | ||||
|             fn name(&self) -> String { | ||||
|                 self.err_code().error_name.to_string() | ||||
|                 match self { | ||||
|                     $( | ||||
|                         Code::$code_ident => stringify!($code_ident).to_case(convert_case::Case::Snake) | ||||
|                     ),* | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             /// return the error type | ||||
|             fn type_(&self) -> String { | ||||
|                 self.err_code().error_type.to_string() | ||||
|                 match self { | ||||
|                     $( | ||||
|                         Code::$code_ident => ErrorType::$err_type.to_string() | ||||
|                     ),* | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             /// return the doc url associated with the error | ||||
| @@ -177,144 +176,121 @@ macro_rules! make_error_codes { | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| // An exhaustive list of all the error codes used by meilisearch. | ||||
| make_error_codes! { | ||||
| ApiKeyAlreadyExists                   , invalid       , CONFLICT ; | ||||
| ApiKeyNotFound                        , invalid       , NOT_FOUND ; | ||||
| BadParameter                          , invalid       , BAD_REQUEST; | ||||
| BadRequest                            , invalid       , BAD_REQUEST; | ||||
| DatabaseSizeLimitReached              , internal      , INTERNAL_SERVER_ERROR; | ||||
| DocumentNotFound                      , invalid       , NOT_FOUND; | ||||
| DumpAlreadyProcessing                 , invalid       , CONFLICT; | ||||
| DumpNotFound                          , invalid       , NOT_FOUND; | ||||
| DumpProcessFailed                     , internal      , INTERNAL_SERVER_ERROR; | ||||
| DuplicateIndexFound                   , invalid       , BAD_REQUEST; | ||||
| ImmutableApiKeyUid                    , invalid       , BAD_REQUEST; | ||||
| ImmutableApiKeyKey                    , invalid       , BAD_REQUEST; | ||||
| ImmutableApiKeyActions                , invalid       , BAD_REQUEST; | ||||
| ImmutableApiKeyIndexes                , invalid       , BAD_REQUEST; | ||||
| ImmutableApiKeyExpiresAt              , invalid       , BAD_REQUEST; | ||||
| ImmutableApiKeyCreatedAt              , invalid       , BAD_REQUEST; | ||||
| ImmutableApiKeyUpdatedAt              , invalid       , BAD_REQUEST; | ||||
| ImmutableIndexUid                     , invalid       , BAD_REQUEST; | ||||
| ImmutableIndexCreatedAt               , invalid       , BAD_REQUEST; | ||||
| ImmutableIndexUpdatedAt               , invalid       , BAD_REQUEST; | ||||
| IndexAlreadyExists                    , invalid       , CONFLICT ; | ||||
| IndexCreationFailed                   , internal      , INTERNAL_SERVER_ERROR; | ||||
| IndexNotFound                         , invalid       , NOT_FOUND; | ||||
| IndexPrimaryKeyAlreadyExists          , invalid       , BAD_REQUEST ; | ||||
| IndexPrimaryKeyNoCandidateFound       , invalid       , BAD_REQUEST ; | ||||
| IndexPrimaryKeyMultipleCandidatesFound, invalid       , BAD_REQUEST; | ||||
| Internal                              , internal      , INTERNAL_SERVER_ERROR ; | ||||
| InvalidApiKeyActions                  , invalid       , BAD_REQUEST ; | ||||
| InvalidApiKeyDescription              , invalid       , BAD_REQUEST ; | ||||
| InvalidApiKeyExpiresAt                , invalid       , BAD_REQUEST ; | ||||
| InvalidApiKeyIndexes                  , invalid       , BAD_REQUEST ; | ||||
| InvalidApiKeyLimit                    , invalid       , BAD_REQUEST ; | ||||
| InvalidApiKeyName                     , invalid       , BAD_REQUEST ; | ||||
| InvalidApiKeyOffset                   , invalid       , BAD_REQUEST ; | ||||
| InvalidApiKeyUid                      , invalid       , BAD_REQUEST ; | ||||
| InvalidApiKey                         , authentication, FORBIDDEN ; | ||||
| InvalidContentType                    , invalid       , UNSUPPORTED_MEDIA_TYPE ; | ||||
| InvalidDocumentFields                 , invalid       , BAD_REQUEST ; | ||||
| InvalidDocumentGeoField               , invalid       , BAD_REQUEST ; | ||||
| InvalidDocumentId                     , invalid       , BAD_REQUEST ; | ||||
| InvalidDocumentLimit                  , invalid       , BAD_REQUEST ; | ||||
| InvalidDocumentOffset                 , invalid       , BAD_REQUEST ; | ||||
| InvalidIndexLimit                     , invalid       , BAD_REQUEST ; | ||||
| InvalidIndexOffset                    , invalid       , BAD_REQUEST ; | ||||
| InvalidIndexPrimaryKey                , invalid       , BAD_REQUEST ; | ||||
| InvalidIndexUid                       , invalid       , BAD_REQUEST ; | ||||
| InvalidMinWordLengthForTypo           , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchAttributesToCrop         , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchAttributesToHighlight    , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchAttributesToRetrieve     , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchCropLength               , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchCropMarker               , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchFacets                   , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchFilter                   , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchHighlightPostTag         , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchHighlightPreTag          , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchHitsPerPage              , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchLimit                    , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchMatchingStrategy         , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchOffset                   , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchPage                     , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchQ                        , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchShowMatchesPosition      , invalid       , BAD_REQUEST ; | ||||
| InvalidSearchSort                     , invalid       , BAD_REQUEST ; | ||||
| InvalidSettingsDisplayedAttributes    , invalid       , BAD_REQUEST ; | ||||
| InvalidSettingsDistinctAttribute      , invalid       , BAD_REQUEST ; | ||||
| InvalidSettingsFaceting               , invalid       , BAD_REQUEST ; | ||||
| InvalidSettingsFilterableAttributes   , invalid       , BAD_REQUEST ; | ||||
| InvalidSettingsPagination             , invalid       , BAD_REQUEST ; | ||||
| InvalidSettingsRankingRules           , invalid       , BAD_REQUEST ; | ||||
| InvalidSettingsSearchableAttributes   , invalid       , BAD_REQUEST ; | ||||
| InvalidSettingsSortableAttributes     , invalid       , BAD_REQUEST ; | ||||
| InvalidSettingsStopWords              , invalid       , BAD_REQUEST ; | ||||
| InvalidSettingsSynonyms               , invalid       , BAD_REQUEST ; | ||||
| InvalidSettingsTypoTolerance          , invalid       , BAD_REQUEST ; | ||||
| InvalidState                          , internal      , INTERNAL_SERVER_ERROR ; | ||||
| InvalidStoreFile                      , internal      , INTERNAL_SERVER_ERROR ; | ||||
| InvalidSwapDuplicateIndexFound        , invalid       , BAD_REQUEST ; | ||||
| InvalidSwapIndexes                    , invalid       , BAD_REQUEST ; | ||||
| InvalidTaskAfterEnqueuedAt            , invalid       , BAD_REQUEST ; | ||||
| InvalidTaskAfterFinishedAt            , invalid       , BAD_REQUEST ; | ||||
| InvalidTaskAfterStartedAt             , invalid       , BAD_REQUEST ; | ||||
| InvalidTaskBeforeEnqueuedAt           , invalid       , BAD_REQUEST ; | ||||
| InvalidTaskBeforeFinishedAt           , invalid       , BAD_REQUEST ; | ||||
| InvalidTaskBeforeStartedAt            , invalid       , BAD_REQUEST ; | ||||
| InvalidTaskCanceledBy                 , invalid       , BAD_REQUEST ; | ||||
| InvalidTaskFrom                       , invalid       , BAD_REQUEST ; | ||||
| InvalidTaskLimit                      , invalid       , BAD_REQUEST ; | ||||
| InvalidTaskStatuses                   , invalid       , BAD_REQUEST ; | ||||
| InvalidTaskTypes                      , invalid       , BAD_REQUEST ; | ||||
| InvalidTaskUids                       , invalid       , BAD_REQUEST  ; | ||||
| IoError                               , system        , UNPROCESSABLE_ENTITY; | ||||
| MalformedPayload                      , invalid       , BAD_REQUEST ; | ||||
| MaxFieldsLimitExceeded                , invalid       , BAD_REQUEST ; | ||||
| MissingApiKeyActions                  , invalid       , BAD_REQUEST ; | ||||
| MissingApiKeyExpiresAt                , invalid       , BAD_REQUEST ; | ||||
| MissingApiKeyIndexes                  , invalid       , BAD_REQUEST ; | ||||
| MissingAuthorizationHeader            , authentication, UNAUTHORIZED ; | ||||
| MissingContentType                    , invalid       , UNSUPPORTED_MEDIA_TYPE ; | ||||
| MissingDocumentId                     , invalid       , BAD_REQUEST ; | ||||
| MissingIndexUid                       , invalid       , BAD_REQUEST ; | ||||
| MissingMasterKey                      , authentication, UNAUTHORIZED ; | ||||
| MissingPayload                        , invalid       , BAD_REQUEST ; | ||||
| MissingSwapIndexes             , invalid       , BAD_REQUEST ; | ||||
| MissingTaskFilters                    , invalid       , BAD_REQUEST ; | ||||
| NoSpaceLeftOnDevice                   , system        , UNPROCESSABLE_ENTITY; | ||||
| PayloadTooLarge                       , invalid       , PAYLOAD_TOO_LARGE ; | ||||
| TaskNotFound                          , invalid       , NOT_FOUND ; | ||||
| TooManyOpenFiles                      , system        , UNPROCESSABLE_ENTITY ; | ||||
| UnretrievableDocument                 , internal      , BAD_REQUEST ; | ||||
| UnretrievableErrorCode                , invalid       , BAD_REQUEST ; | ||||
| UnsupportedMediaType                  , invalid       , UNSUPPORTED_MEDIA_TYPE | ||||
| } | ||||
|  | ||||
| /// Internal structure providing a convenient way to create error codes | ||||
| struct ErrCode { | ||||
|     status_code: StatusCode, | ||||
|     error_type: ErrorType, | ||||
|     error_name: String, | ||||
| } | ||||
|  | ||||
| impl ErrCode { | ||||
|     fn authentication(error_name: String, status_code: StatusCode) -> ErrCode { | ||||
|         ErrCode { status_code, error_name, error_type: ErrorType::AuthenticationError } | ||||
|     } | ||||
|  | ||||
|     fn internal(error_name: String, status_code: StatusCode) -> ErrCode { | ||||
|         ErrCode { status_code, error_name, error_type: ErrorType::InternalError } | ||||
|     } | ||||
|  | ||||
|     fn invalid(error_name: String, status_code: StatusCode) -> ErrCode { | ||||
|         ErrCode { status_code, error_name, error_type: ErrorType::InvalidRequestError } | ||||
|     } | ||||
|  | ||||
|     fn system(error_name: String, status_code: StatusCode) -> ErrCode { | ||||
|         ErrCode { status_code, error_name, error_type: ErrorType::System } | ||||
|     } | ||||
| ApiKeyAlreadyExists                   , InvalidRequest       , CONFLICT ; | ||||
| ApiKeyNotFound                        , InvalidRequest       , NOT_FOUND ; | ||||
| BadParameter                          , InvalidRequest       , BAD_REQUEST; | ||||
| BadRequest                            , InvalidRequest       , BAD_REQUEST; | ||||
| DatabaseSizeLimitReached              , Internal             , INTERNAL_SERVER_ERROR; | ||||
| DocumentNotFound                      , InvalidRequest       , NOT_FOUND; | ||||
| DumpAlreadyProcessing                 , InvalidRequest       , CONFLICT; | ||||
| DumpNotFound                          , InvalidRequest       , NOT_FOUND; | ||||
| DumpProcessFailed                     , Internal             , INTERNAL_SERVER_ERROR; | ||||
| DuplicateIndexFound                   , InvalidRequest       , BAD_REQUEST; | ||||
| ImmutableApiKeyActions                , InvalidRequest       , BAD_REQUEST; | ||||
| ImmutableApiKeyCreatedAt              , InvalidRequest       , BAD_REQUEST; | ||||
| ImmutableApiKeyExpiresAt              , InvalidRequest       , BAD_REQUEST; | ||||
| ImmutableApiKeyIndexes                , InvalidRequest       , BAD_REQUEST; | ||||
| ImmutableApiKeyKey                    , InvalidRequest       , BAD_REQUEST; | ||||
| ImmutableApiKeyUid                    , InvalidRequest       , BAD_REQUEST; | ||||
| ImmutableApiKeyUpdatedAt              , InvalidRequest       , BAD_REQUEST; | ||||
| ImmutableIndexCreatedAt               , InvalidRequest       , BAD_REQUEST; | ||||
| ImmutableIndexUid                     , InvalidRequest       , BAD_REQUEST; | ||||
| ImmutableIndexUpdatedAt               , InvalidRequest       , BAD_REQUEST; | ||||
| IndexAlreadyExists                    , InvalidRequest       , CONFLICT ; | ||||
| IndexCreationFailed                   , Internal             , INTERNAL_SERVER_ERROR; | ||||
| IndexNotFound                         , InvalidRequest       , NOT_FOUND; | ||||
| IndexPrimaryKeyAlreadyExists          , InvalidRequest       , BAD_REQUEST ; | ||||
| IndexPrimaryKeyMultipleCandidatesFound, InvalidRequest       , BAD_REQUEST; | ||||
| IndexPrimaryKeyNoCandidateFound       , InvalidRequest       , BAD_REQUEST ; | ||||
| Internal                              , Internal             , INTERNAL_SERVER_ERROR ; | ||||
| InvalidApiKey                         , Auth                 , FORBIDDEN ; | ||||
| InvalidApiKeyActions                  , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidApiKeyDescription              , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidApiKeyExpiresAt                , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidApiKeyIndexes                  , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidApiKeyLimit                    , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidApiKeyName                     , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidApiKeyOffset                   , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidApiKeyUid                      , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidContentType                    , InvalidRequest       , UNSUPPORTED_MEDIA_TYPE ; | ||||
| InvalidDocumentFields                 , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidDocumentGeoField               , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidDocumentId                     , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidDocumentLimit                  , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidDocumentOffset                 , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidIndexLimit                     , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidIndexOffset                    , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidIndexPrimaryKey                , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidIndexUid                       , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidMinWordLengthForTypo           , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchAttributesToCrop         , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchAttributesToHighlight    , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchAttributesToRetrieve     , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchCropLength               , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchCropMarker               , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchFacets                   , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchFilter                   , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchHighlightPostTag         , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchHighlightPreTag          , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchHitsPerPage              , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchLimit                    , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchMatchingStrategy         , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchOffset                   , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchPage                     , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchQ                        , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchShowMatchesPosition      , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSearchSort                     , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSettingsDisplayedAttributes    , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSettingsDistinctAttribute      , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSettingsFaceting               , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSettingsFilterableAttributes   , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSettingsPagination             , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSettingsRankingRules           , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSettingsSearchableAttributes   , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSettingsSortableAttributes     , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSettingsStopWords              , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSettingsSynonyms               , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSettingsTypoTolerance          , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidState                          , Internal             , INTERNAL_SERVER_ERROR ; | ||||
| InvalidStoreFile                      , Internal             , INTERNAL_SERVER_ERROR ; | ||||
| InvalidSwapDuplicateIndexFound        , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidSwapIndexes                    , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidTaskAfterEnqueuedAt            , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidTaskAfterFinishedAt            , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidTaskAfterStartedAt             , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidTaskBeforeEnqueuedAt           , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidTaskBeforeFinishedAt           , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidTaskBeforeStartedAt            , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidTaskCanceledBy                 , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidTaskFrom                       , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidTaskLimit                      , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidTaskStatuses                   , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidTaskTypes                      , InvalidRequest       , BAD_REQUEST ; | ||||
| InvalidTaskUids                       , InvalidRequest       , BAD_REQUEST  ; | ||||
| IoError                               , System               , UNPROCESSABLE_ENTITY; | ||||
| MalformedPayload                      , InvalidRequest       , BAD_REQUEST ; | ||||
| MaxFieldsLimitExceeded                , InvalidRequest       , BAD_REQUEST ; | ||||
| MissingApiKeyActions                  , InvalidRequest       , BAD_REQUEST ; | ||||
| MissingApiKeyExpiresAt                , InvalidRequest       , BAD_REQUEST ; | ||||
| MissingApiKeyIndexes                  , InvalidRequest       , BAD_REQUEST ; | ||||
| MissingAuthorizationHeader            , Auth                 , UNAUTHORIZED ; | ||||
| MissingContentType                    , InvalidRequest       , UNSUPPORTED_MEDIA_TYPE ; | ||||
| MissingDocumentId                     , InvalidRequest       , BAD_REQUEST ; | ||||
| MissingIndexUid                       , InvalidRequest       , BAD_REQUEST ; | ||||
| MissingMasterKey                      , Auth                 , UNAUTHORIZED ; | ||||
| MissingPayload                        , InvalidRequest       , BAD_REQUEST ; | ||||
| MissingSwapIndexes                    , InvalidRequest       , BAD_REQUEST ; | ||||
| MissingTaskFilters                    , InvalidRequest       , BAD_REQUEST ; | ||||
| NoSpaceLeftOnDevice                   , System               , UNPROCESSABLE_ENTITY; | ||||
| PayloadTooLarge                       , InvalidRequest       , PAYLOAD_TOO_LARGE ; | ||||
| TaskNotFound                          , InvalidRequest       , NOT_FOUND ; | ||||
| TooManyOpenFiles                      , System               , UNPROCESSABLE_ENTITY ; | ||||
| UnretrievableDocument                 , Internal             , BAD_REQUEST ; | ||||
| UnretrievableErrorCode                , InvalidRequest       , BAD_REQUEST ; | ||||
| UnsupportedMediaType                  , InvalidRequest       , UNSUPPORTED_MEDIA_TYPE | ||||
| } | ||||
|  | ||||
| impl ErrorCode for JoinError { | ||||
| @@ -409,6 +385,7 @@ impl ErrorCode for io::Error { | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Unwrap a result, either its Ok or Err value. | ||||
| pub fn unwrap_any<T>(any: Result<T, T>) -> T { | ||||
|     match any { | ||||
|         Ok(any) => any, | ||||
| @@ -416,501 +393,43 @@ pub fn unwrap_any<T>(any: Result<T, T>) -> T { | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg(feature = "test-traits")] | ||||
| mod strategy { | ||||
|     use proptest::strategy::Strategy; | ||||
|  | ||||
|     use super::*; | ||||
|  | ||||
|     pub(super) fn status_code_strategy() -> impl Strategy<Value = StatusCode> { | ||||
|         (100..999u16).prop_map(|i| StatusCode::from_u16(i).unwrap()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub struct DeserrJson; | ||||
| pub struct DeserrQueryParam; | ||||
|  | ||||
| pub type DeserrJsonError<C = deserr_codes::BadRequest> = DeserrError<DeserrJson, C>; | ||||
| pub type DeserrQueryParamError<C = deserr_codes::BadRequest> = DeserrError<DeserrQueryParam, C>; | ||||
|  | ||||
| pub struct DeserrError<Format, C: Default + ErrorCode> { | ||||
|     pub msg: String, | ||||
|     pub code: Code, | ||||
|     _phantom: PhantomData<(Format, C)>, | ||||
| } | ||||
| impl<Format, C: Default + ErrorCode> std::fmt::Debug for DeserrError<Format, C> { | ||||
|     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||||
|         f.debug_struct("DeserrError").field("msg", &self.msg).field("code", &self.code).finish() | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<Format, C: Default + ErrorCode> std::fmt::Display for DeserrError<Format, C> { | ||||
|     fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||||
|         write!(f, "{}", self.msg) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<Format, C: Default + ErrorCode> std::error::Error for DeserrError<Format, C> {} | ||||
| impl<Format, C: Default + ErrorCode> ErrorCode for DeserrError<Format, C> { | ||||
|     fn error_code(&self) -> Code { | ||||
|         self.code | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<Format, C1: Default + ErrorCode, C2: Default + ErrorCode> | ||||
|     MergeWithError<DeserrError<Format, C2>> for DeserrError<Format, C1> | ||||
| { | ||||
|     fn merge( | ||||
|         _self_: Option<Self>, | ||||
|         other: DeserrError<Format, C2>, | ||||
|         _merge_location: ValuePointerRef, | ||||
|     ) -> Result<Self, Self> { | ||||
|         Err(DeserrError { msg: other.msg, code: other.code, _phantom: PhantomData }) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl DeserrJsonError<MissingIndexUid> { | ||||
|     pub fn missing_index_uid(field: &str, location: ValuePointerRef) -> Self { | ||||
|         let x = unwrap_any(Self::error::<Infallible>( | ||||
|             None, | ||||
|             deserr::ErrorKind::MissingField { field }, | ||||
|             location, | ||||
|         )); | ||||
|         Self { msg: x.msg, code: MissingIndexUid.error_code(), _phantom: PhantomData } | ||||
|     } | ||||
| } | ||||
| impl DeserrJsonError<MissingApiKeyActions> { | ||||
|     pub fn missing_api_key_actions(field: &str, location: ValuePointerRef) -> Self { | ||||
|         let x = unwrap_any(Self::error::<Infallible>( | ||||
|             None, | ||||
|             deserr::ErrorKind::MissingField { field }, | ||||
|             location, | ||||
|         )); | ||||
|         Self { msg: x.msg, code: MissingApiKeyActions.error_code(), _phantom: PhantomData } | ||||
|     } | ||||
| } | ||||
| impl DeserrJsonError<MissingApiKeyExpiresAt> { | ||||
|     pub fn missing_api_key_expires_at(field: &str, location: ValuePointerRef) -> Self { | ||||
|         let x = unwrap_any(Self::error::<Infallible>( | ||||
|             None, | ||||
|             deserr::ErrorKind::MissingField { field }, | ||||
|             location, | ||||
|         )); | ||||
|         Self { msg: x.msg, code: MissingApiKeyExpiresAt.error_code(), _phantom: PhantomData } | ||||
|     } | ||||
| } | ||||
| impl DeserrJsonError<MissingApiKeyIndexes> { | ||||
|     pub fn missing_api_key_indexes(field: &str, location: ValuePointerRef) -> Self { | ||||
|         let x = unwrap_any(Self::error::<Infallible>( | ||||
|             None, | ||||
|             deserr::ErrorKind::MissingField { field }, | ||||
|             location, | ||||
|         )); | ||||
|         Self { msg: x.msg, code: MissingApiKeyIndexes.error_code(), _phantom: PhantomData } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl DeserrJsonError<InvalidSwapIndexes> { | ||||
|     pub fn missing_swap_indexes_indexes(field: &str, location: ValuePointerRef) -> Self { | ||||
|         let x = unwrap_any(Self::error::<Infallible>( | ||||
|             None, | ||||
|             deserr::ErrorKind::MissingField { field }, | ||||
|             location, | ||||
|         )); | ||||
|         Self { msg: x.msg, code: MissingSwapIndexes.error_code(), _phantom: PhantomData } | ||||
|     } | ||||
| } | ||||
|  | ||||
| // if the error happened in the root, then an empty string is returned. | ||||
| pub fn location_json_description(location: ValuePointerRef, article: &str) -> String { | ||||
|     fn rec(location: ValuePointerRef) -> String { | ||||
|         match location { | ||||
|             ValuePointerRef::Origin => String::new(), | ||||
|             ValuePointerRef::Key { key, prev } => rec(*prev) + "." + key, | ||||
|             ValuePointerRef::Index { index, prev } => format!("{}[{index}]", rec(*prev)), | ||||
|         } | ||||
|     } | ||||
|     match location { | ||||
|         ValuePointerRef::Origin => String::new(), | ||||
|         _ => { | ||||
|             format!("{article} `{}`", rec(location)) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn value_kinds_description_json(kinds: &[ValueKind]) -> String { | ||||
|     fn order(kind: &ValueKind) -> u8 { | ||||
|         match kind { | ||||
|             ValueKind::Null => 0, | ||||
|             ValueKind::Boolean => 1, | ||||
|             ValueKind::Integer => 2, | ||||
|             ValueKind::NegativeInteger => 3, | ||||
|             ValueKind::Float => 4, | ||||
|             ValueKind::String => 5, | ||||
|             ValueKind::Sequence => 6, | ||||
|             ValueKind::Map => 7, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn single_description(kind: &ValueKind) -> &'static str { | ||||
|         match kind { | ||||
|             ValueKind::Null => "null", | ||||
|             ValueKind::Boolean => "a boolean", | ||||
|             ValueKind::Integer => "a positive integer", | ||||
|             ValueKind::NegativeInteger => "an integer", | ||||
|             ValueKind::Float => "a number", | ||||
|             ValueKind::String => "a string", | ||||
|             ValueKind::Sequence => "an array", | ||||
|             ValueKind::Map => "an object", | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn description_rec(kinds: &[ValueKind], count_items: &mut usize, message: &mut String) { | ||||
|         let (msg_part, rest): (_, &[ValueKind]) = match kinds { | ||||
|             [] => (String::new(), &[]), | ||||
|             [ValueKind::Integer | ValueKind::NegativeInteger, ValueKind::Float, rest @ ..] => { | ||||
|                 ("a number".to_owned(), rest) | ||||
|             } | ||||
|             [ValueKind::Integer, ValueKind::NegativeInteger, ValueKind::Float, rest @ ..] => { | ||||
|                 ("a number".to_owned(), rest) | ||||
|             } | ||||
|             [ValueKind::Integer, ValueKind::NegativeInteger, rest @ ..] => { | ||||
|                 ("an integer".to_owned(), rest) | ||||
|             } | ||||
|             [a] => (single_description(a).to_owned(), &[]), | ||||
|             [a, rest @ ..] => (single_description(a).to_owned(), rest), | ||||
|         }; | ||||
|  | ||||
|         if rest.is_empty() { | ||||
|             if *count_items == 0 { | ||||
|                 message.push_str(&msg_part); | ||||
|             } else if *count_items == 1 { | ||||
|                 message.push_str(&format!(" or {msg_part}")); | ||||
|             } else { | ||||
|                 message.push_str(&format!(", or {msg_part}")); | ||||
|             } | ||||
|         } else { | ||||
|             if *count_items == 0 { | ||||
|                 message.push_str(&msg_part); | ||||
|             } else { | ||||
|                 message.push_str(&format!(", {msg_part}")); | ||||
|             } | ||||
|  | ||||
|             *count_items += 1; | ||||
|             description_rec(rest, count_items, message); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     let mut kinds = kinds.to_owned(); | ||||
|     kinds.sort_by_key(order); | ||||
|     kinds.dedup(); | ||||
|  | ||||
|     if kinds.is_empty() { | ||||
|         "a different value".to_owned() | ||||
|     } else { | ||||
|         let mut message = String::new(); | ||||
|         description_rec(kinds.as_slice(), &mut 0, &mut message); | ||||
|         message | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn value_description_with_kind_json(v: &serde_json::Value) -> String { | ||||
|     match v.kind() { | ||||
|         ValueKind::Null => "null".to_owned(), | ||||
|         kind => { | ||||
|             format!( | ||||
|                 "{}: `{}`", | ||||
|                 value_kinds_description_json(&[kind]), | ||||
|                 serde_json::to_string(v).unwrap() | ||||
|             ) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<C: Default + ErrorCode> deserr::DeserializeError for DeserrJsonError<C> { | ||||
|     fn error<V: IntoValue>( | ||||
|         _self_: Option<Self>, | ||||
|         error: deserr::ErrorKind<V>, | ||||
|         location: ValuePointerRef, | ||||
|     ) -> Result<Self, Self> { | ||||
|         let mut message = String::new(); | ||||
|  | ||||
|         message.push_str(&match error { | ||||
|             ErrorKind::IncorrectValueKind { actual, accepted } => { | ||||
|                 let expected = value_kinds_description_json(accepted); | ||||
|                 // if we're not able to get the value as a string then we print nothing. | ||||
|                 let received = value_description_with_kind_json(&serde_json::Value::from(actual)); | ||||
|  | ||||
|                 let location = location_json_description(location, " at"); | ||||
|  | ||||
|                 format!("Invalid value type{location}: expected {expected}, but found {received}") | ||||
|             } | ||||
|             ErrorKind::MissingField { field } => { | ||||
|                 // serde_json original message: | ||||
|                 // Json deserialize error: missing field `lol` at line 1 column 2 | ||||
|                 let location = location_json_description(location, " inside"); | ||||
|                 format!("Missing field `{field}`{location}") | ||||
|             } | ||||
|             ErrorKind::UnknownKey { key, accepted } => { | ||||
|                 let location = location_json_description(location, " inside"); | ||||
|                 format!( | ||||
|                     "Unknown field `{}`{location}: expected one of {}", | ||||
|                     key, | ||||
|                     accepted | ||||
|                         .iter() | ||||
|                         .map(|accepted| format!("`{}`", accepted)) | ||||
|                         .collect::<Vec<String>>() | ||||
|                         .join(", ") | ||||
|                 ) | ||||
|             } | ||||
|             ErrorKind::UnknownValue { value, accepted } => { | ||||
|                 let location = location_json_description(location, " at"); | ||||
|                 format!( | ||||
|                     "Unknown value `{}`{location}: expected one of {}", | ||||
|                     value, | ||||
|                     accepted | ||||
|                         .iter() | ||||
|                         .map(|accepted| format!("`{}`", accepted)) | ||||
|                         .collect::<Vec<String>>() | ||||
|                         .join(", "), | ||||
|                 ) | ||||
|             } | ||||
|             ErrorKind::Unexpected { msg } => { | ||||
|                 let location = location_json_description(location, " at"); | ||||
|                 // serde_json original message: | ||||
|                 // The json payload provided is malformed. `trailing characters at line 1 column 19`. | ||||
|                 format!("Invalid value{location}: {msg}") | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         Err(DeserrJsonError { | ||||
|             msg: message, | ||||
|             code: C::default().error_code(), | ||||
|             _phantom: PhantomData, | ||||
|         }) | ||||
|     } | ||||
| } | ||||
|  | ||||
| // if the error happened in the root, then an empty string is returned. | ||||
| pub fn location_query_param_description(location: ValuePointerRef, article: &str) -> String { | ||||
|     fn rec(location: ValuePointerRef) -> String { | ||||
|         match location { | ||||
|             ValuePointerRef::Origin => String::new(), | ||||
|             ValuePointerRef::Key { key, prev } => { | ||||
|                 if matches!(prev, ValuePointerRef::Origin) { | ||||
|                     key.to_owned() | ||||
|                 } else { | ||||
|                     rec(*prev) + "." + key | ||||
|                 } | ||||
|             } | ||||
|             ValuePointerRef::Index { index, prev } => format!("{}[{index}]", rec(*prev)), | ||||
|         } | ||||
|     } | ||||
|     match location { | ||||
|         ValuePointerRef::Origin => String::new(), | ||||
|         _ => { | ||||
|             format!("{article} `{}`", rec(location)) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<C: Default + ErrorCode> deserr::DeserializeError for DeserrQueryParamError<C> { | ||||
|     fn error<V: IntoValue>( | ||||
|         _self_: Option<Self>, | ||||
|         error: deserr::ErrorKind<V>, | ||||
|         location: ValuePointerRef, | ||||
|     ) -> Result<Self, Self> { | ||||
|         let mut message = String::new(); | ||||
|  | ||||
|         message.push_str(&match error { | ||||
|             ErrorKind::IncorrectValueKind { actual, accepted } => { | ||||
|                 let expected = value_kinds_description_query_param(accepted); | ||||
|                 // if we're not able to get the value as a string then we print nothing. | ||||
|                 let received = value_description_with_kind_query_param(actual); | ||||
|  | ||||
|                 let location = location_query_param_description(location, " for parameter"); | ||||
|  | ||||
|                 format!("Invalid value type{location}: expected {expected}, but found {received}") | ||||
|             } | ||||
|             ErrorKind::MissingField { field } => { | ||||
|                 // serde_json original message: | ||||
|                 // Json deserialize error: missing field `lol` at line 1 column 2 | ||||
|                 let location = location_query_param_description(location, " inside"); | ||||
|                 format!("Missing parameter `{field}`{location}") | ||||
|             } | ||||
|             ErrorKind::UnknownKey { key, accepted } => { | ||||
|                 let location = location_query_param_description(location, " inside"); | ||||
|                 format!( | ||||
|                     "Unknown parameter `{}`{location}: expected one of {}", | ||||
|                     key, | ||||
|                     accepted | ||||
|                         .iter() | ||||
|                         .map(|accepted| format!("`{}`", accepted)) | ||||
|                         .collect::<Vec<String>>() | ||||
|                         .join(", ") | ||||
|                 ) | ||||
|             } | ||||
|             ErrorKind::UnknownValue { value, accepted } => { | ||||
|                 let location = location_query_param_description(location, " for parameter"); | ||||
|                 format!( | ||||
|                     "Unknown value `{}`{location}: expected one of {}", | ||||
|                     value, | ||||
|                     accepted | ||||
|                         .iter() | ||||
|                         .map(|accepted| format!("`{}`", accepted)) | ||||
|                         .collect::<Vec<String>>() | ||||
|                         .join(", "), | ||||
|                 ) | ||||
|             } | ||||
|             ErrorKind::Unexpected { msg } => { | ||||
|                 let location = location_query_param_description(location, " in parameter"); | ||||
|                 // serde_json original message: | ||||
|                 // The json payload provided is malformed. `trailing characters at line 1 column 19`. | ||||
|                 format!("Invalid value{location}: {msg}") | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         Err(DeserrQueryParamError { | ||||
|             msg: message, | ||||
|             code: C::default().error_code(), | ||||
|             _phantom: PhantomData, | ||||
|         }) | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn value_kinds_description_query_param(_accepted: &[ValueKind]) -> String { | ||||
|     "a string".to_owned() | ||||
| } | ||||
|  | ||||
| fn value_description_with_kind_query_param<V: IntoValue>(actual: deserr::Value<V>) -> String { | ||||
|     match actual { | ||||
|         deserr::Value::Null => "null".to_owned(), | ||||
|         deserr::Value::Boolean(x) => format!("a boolean: `{x}`"), | ||||
|         deserr::Value::Integer(x) => format!("an integer: `{x}`"), | ||||
|         deserr::Value::NegativeInteger(x) => { | ||||
|             format!("an integer: `{x}`") | ||||
|         } | ||||
|         deserr::Value::Float(x) => { | ||||
|             format!("a number: `{x}`") | ||||
|         } | ||||
|         deserr::Value::String(x) => { | ||||
|             format!("a string: `{x}`") | ||||
|         } | ||||
|         deserr::Value::Sequence(_) => "multiple values".to_owned(), | ||||
|         deserr::Value::Map(_) => "multiple parameters".to_owned(), | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Deserialization when `deserr` cannot parse an API key date. | ||||
| #[derive(Debug)] | ||||
| pub struct DetailedParseIntError(String); | ||||
| impl fmt::Display for DetailedParseIntError { | ||||
| pub struct ParseOffsetDateTimeError(pub String); | ||||
| impl fmt::Display for ParseOffsetDateTimeError { | ||||
|     fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||||
|         writeln!(f, "`{original}` is not a valid date. It should follow the RFC 3339 format to represents a date or datetime in the future or specified as a null value. e.g. 'YYYY-MM-DD' or 'YYYY-MM-DD HH:MM:SS'.", original = self.0) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Deserialization when `deserr` cannot parse a task date. | ||||
| #[derive(Debug)] | ||||
| pub struct InvalidTaskDateError(pub String); | ||||
| impl std::fmt::Display for InvalidTaskDateError { | ||||
|     fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||||
|         write!(f, "`{}` is an invalid date-time. It should follow the YYYY-MM-DD or RFC 3339 date-time format.", self.0) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Deserialization error when `deserr` cannot parse a String | ||||
| /// into a bool. | ||||
| #[derive(Debug)] | ||||
| pub struct DeserrParseBoolError(pub String); | ||||
| impl fmt::Display for DeserrParseBoolError { | ||||
|     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||||
|         write!(f, "could not parse `{}` as a boolean, expected either `true` or `false`", self.0) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Deserialization error when `deserr` cannot parse a String | ||||
| /// into an integer. | ||||
| #[derive(Debug)] | ||||
| pub struct DeserrParseIntError(pub String); | ||||
| impl fmt::Display for DeserrParseIntError { | ||||
|     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||||
|         write!(f, "could not parse `{}` as a positive integer", self.0) | ||||
|     } | ||||
| } | ||||
| impl std::error::Error for DetailedParseIntError {} | ||||
|  | ||||
| pub fn parse_u32_query_param(x: String) -> Result<u32, TakeErrorMessage<DetailedParseIntError>> { | ||||
|     x.parse::<u32>().map_err(|_e| TakeErrorMessage(DetailedParseIntError(x.to_owned()))) | ||||
| } | ||||
| pub fn parse_usize_query_param( | ||||
|     x: String, | ||||
| ) -> Result<usize, TakeErrorMessage<DetailedParseIntError>> { | ||||
|     x.parse::<usize>().map_err(|_e| TakeErrorMessage(DetailedParseIntError(x.to_owned()))) | ||||
| } | ||||
| pub fn parse_option_usize_query_param( | ||||
|     s: Option<String>, | ||||
| ) -> Result<Option<usize>, TakeErrorMessage<DetailedParseIntError>> { | ||||
|     if let Some(s) = s { | ||||
|         parse_usize_query_param(s).map(Some) | ||||
|     } else { | ||||
|         Ok(None) | ||||
|     } | ||||
| } | ||||
| pub fn parse_option_u32_query_param( | ||||
|     s: Option<String>, | ||||
| ) -> Result<Option<u32>, TakeErrorMessage<DetailedParseIntError>> { | ||||
|     if let Some(s) = s { | ||||
|         parse_u32_query_param(s).map(Some) | ||||
|     } else { | ||||
|         Ok(None) | ||||
|     } | ||||
| } | ||||
| pub fn parse_option_vec_u32_query_param( | ||||
|     s: Option<serde_cs::vec::CS<String>>, | ||||
| ) -> Result<Option<Vec<u32>>, TakeErrorMessage<DetailedParseIntError>> { | ||||
|     if let Some(s) = s { | ||||
|         s.into_iter() | ||||
|             .map(parse_u32_query_param) | ||||
|             .collect::<Result<Vec<u32>, TakeErrorMessage<DetailedParseIntError>>>() | ||||
|             .map(Some) | ||||
|     } else { | ||||
|         Ok(None) | ||||
|     } | ||||
| } | ||||
| pub fn parse_option_cs_star_or<T: FromStr>( | ||||
|     s: Option<CS<StarOr<String>>>, | ||||
| ) -> Result<Option<Vec<T>>, TakeErrorMessage<T::Err>> { | ||||
|     if let Some(s) = s.and_then(fold_star_or) as Option<Vec<String>> { | ||||
|         s.into_iter() | ||||
|             .map(|s| T::from_str(&s)) | ||||
|             .collect::<Result<Vec<T>, T::Err>>() | ||||
|             .map_err(TakeErrorMessage) | ||||
|             .map(Some) | ||||
|     } else { | ||||
|         Ok(None) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Extracts the raw values from the `StarOr` types and | ||||
| /// return None if a `StarOr::Star` is encountered. | ||||
| pub fn fold_star_or<T, O>(content: impl IntoIterator<Item = StarOr<T>>) -> Option<O> | ||||
| where | ||||
|     O: FromIterator<T>, | ||||
| { | ||||
|     content | ||||
|         .into_iter() | ||||
|         .map(|value| match value { | ||||
|             StarOr::Star => None, | ||||
|             StarOr::Other(val) => Some(val), | ||||
|         }) | ||||
|         .collect() | ||||
| } | ||||
| pub struct TakeErrorMessage<T>(pub T); | ||||
|  | ||||
| impl<C: Default + ErrorCode, T> MergeWithError<TakeErrorMessage<T>> for DeserrJsonError<C> | ||||
| where | ||||
|     T: std::error::Error, | ||||
| { | ||||
|     fn merge( | ||||
|         _self_: Option<Self>, | ||||
|         other: TakeErrorMessage<T>, | ||||
|         merge_location: ValuePointerRef, | ||||
|     ) -> Result<Self, Self> { | ||||
|         DeserrJsonError::error::<Infallible>( | ||||
|             None, | ||||
|             deserr::ErrorKind::Unexpected { msg: other.0.to_string() }, | ||||
|             merge_location, | ||||
|         ) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<C: Default + ErrorCode, T> MergeWithError<TakeErrorMessage<T>> for DeserrQueryParamError<C> | ||||
| where | ||||
|     T: std::error::Error, | ||||
| { | ||||
|     fn merge( | ||||
|         _self_: Option<Self>, | ||||
|         other: TakeErrorMessage<T>, | ||||
|         merge_location: ValuePointerRef, | ||||
|     ) -> Result<Self, Self> { | ||||
|         DeserrQueryParamError::error::<Infallible>( | ||||
|             None, | ||||
|             deserr::ErrorKind::Unexpected { msg: other.0.to_string() }, | ||||
|             merge_location, | ||||
|         ) | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[macro_export] | ||||
| macro_rules! internal_error { | ||||
| @@ -924,32 +443,3 @@ macro_rules! internal_error { | ||||
|         )* | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg(test)] | ||||
| mod tests { | ||||
|     use deserr::ValueKind; | ||||
|  | ||||
|     use crate::error::value_kinds_description_json; | ||||
|  | ||||
|     #[test] | ||||
|     fn test_value_kinds_description_json() { | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[]), @"a different value"); | ||||
|  | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Boolean]), @"a boolean"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Integer]), @"a positive integer"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::NegativeInteger]), @"an integer"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Integer]), @"a positive integer"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::String]), @"a string"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Sequence]), @"an array"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Map]), @"an object"); | ||||
|  | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Integer, ValueKind::Boolean]), @"a boolean or a positive integer"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Null, ValueKind::Integer]), @"null or a positive integer"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Sequence, ValueKind::NegativeInteger]), @"an integer or an array"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Integer, ValueKind::Float]), @"a number"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Integer, ValueKind::Float, ValueKind::NegativeInteger]), @"a number"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Integer, ValueKind::Float, ValueKind::NegativeInteger, ValueKind::Null]), @"null or a number"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Boolean, ValueKind::Integer, ValueKind::Float, ValueKind::NegativeInteger, ValueKind::Null]), @"null, a boolean, or a number"); | ||||
|         insta::assert_display_snapshot!(value_kinds_description_json(&[ValueKind::Null, ValueKind::Boolean, ValueKind::Integer, ValueKind::Float, ValueKind::NegativeInteger, ValueKind::Null]), @"null, a boolean, or a number"); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -29,6 +29,12 @@ impl IndexUid { | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl fmt::Display for IndexUid { | ||||
|     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||||
|         fmt::Display::fmt(&self.0, f) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl std::ops::Deref for IndexUid { | ||||
|     type Target = str; | ||||
|  | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| use std::convert::Infallible; | ||||
| use std::fmt::Display; | ||||
| use std::hash::Hash; | ||||
| use std::str::FromStr; | ||||
|  | ||||
| use deserr::{DeserializeError, DeserializeFromValue, MergeWithError, ValuePointerRef}; | ||||
| use deserr::{DeserializeError, DeserializeFromValue, ValuePointerRef}; | ||||
| use enum_iterator::Sequence; | ||||
| use serde::{Deserialize, Serialize}; | ||||
| use time::format_description::well_known::Rfc3339; | ||||
| @@ -10,31 +10,14 @@ use time::macros::{format_description, time}; | ||||
| use time::{Date, OffsetDateTime, PrimitiveDateTime}; | ||||
| use uuid::Uuid; | ||||
|  | ||||
| use crate::error::deserr_codes::*; | ||||
| use crate::error::{unwrap_any, Code, DeserrJsonError, ErrorCode, TakeErrorMessage}; | ||||
| use crate::index_uid::{IndexUid, IndexUidFormatError}; | ||||
| use crate::deserr::DeserrJsonError; | ||||
| use crate::error::{deserr_codes::*, ParseOffsetDateTimeError}; | ||||
| use crate::error::{unwrap_any, Code}; | ||||
| use crate::index_uid::IndexUid; | ||||
| use crate::star_or::StarOr; | ||||
|  | ||||
| pub type KeyId = Uuid; | ||||
|  | ||||
| impl<C: Default + ErrorCode> MergeWithError<IndexUidFormatError> for DeserrJsonError<C> { | ||||
|     fn merge( | ||||
|         _self_: Option<Self>, | ||||
|         other: IndexUidFormatError, | ||||
|         merge_location: deserr::ValuePointerRef, | ||||
|     ) -> std::result::Result<Self, Self> { | ||||
|         DeserrJsonError::error::<Infallible>( | ||||
|             None, | ||||
|             deserr::ErrorKind::Unexpected { msg: other.to_string() }, | ||||
|             merge_location, | ||||
|         ) | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn parse_uuid_from_str(s: &str) -> Result<Uuid, TakeErrorMessage<uuid::Error>> { | ||||
|     Uuid::parse_str(s).map_err(TakeErrorMessage) | ||||
| } | ||||
|  | ||||
| #[derive(Debug, DeserializeFromValue)] | ||||
| #[deserr(error = DeserrJsonError, rename_all = camelCase, deny_unknown_fields)] | ||||
| pub struct CreateApiKey { | ||||
| @@ -42,13 +25,13 @@ pub struct CreateApiKey { | ||||
|     pub description: Option<String>, | ||||
|     #[deserr(default, error = DeserrJsonError<InvalidApiKeyName>)] | ||||
|     pub name: Option<String>, | ||||
|     #[deserr(default = Uuid::new_v4(), error = DeserrJsonError<InvalidApiKeyUid>, from(&String) = parse_uuid_from_str -> TakeErrorMessage<uuid::Error>)] | ||||
|     #[deserr(default = Uuid::new_v4(), error = DeserrJsonError<InvalidApiKeyUid>, from(&String) = Uuid::from_str -> uuid::Error)] | ||||
|     pub uid: KeyId, | ||||
|     #[deserr(error = DeserrJsonError<InvalidApiKeyActions>, missing_field_error = DeserrJsonError::missing_api_key_actions)] | ||||
|     pub actions: Vec<Action>, | ||||
|     #[deserr(error = DeserrJsonError<InvalidApiKeyIndexes>, missing_field_error = DeserrJsonError::missing_api_key_indexes)] | ||||
|     pub indexes: Vec<StarOr<IndexUid>>, | ||||
|     #[deserr(error = DeserrJsonError<InvalidApiKeyExpiresAt>, from(Option<String>) = parse_expiration_date -> TakeErrorMessage<ParseOffsetDateTimeError>, missing_field_error = DeserrJsonError::missing_api_key_expires_at)] | ||||
|     #[deserr(error = DeserrJsonError<InvalidApiKeyExpiresAt>, from(Option<String>) = parse_expiration_date -> ParseOffsetDateTimeError, missing_field_error = DeserrJsonError::missing_api_key_expires_at)] | ||||
|     pub expires_at: Option<OffsetDateTime>, | ||||
| } | ||||
| impl CreateApiKey { | ||||
| @@ -149,18 +132,9 @@ impl Key { | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Debug)] | ||||
| pub struct ParseOffsetDateTimeError(String); | ||||
| impl Display for ParseOffsetDateTimeError { | ||||
|     fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||||
|         writeln!(f, "`{original}` is not a valid date. It should follow the RFC 3339 format to represents a date or datetime in the future or specified as a null value. e.g. 'YYYY-MM-DD' or 'YYYY-MM-DD HH:MM:SS'.", original = self.0) | ||||
|     } | ||||
| } | ||||
| impl std::error::Error for ParseOffsetDateTimeError {} | ||||
|  | ||||
| fn parse_expiration_date( | ||||
|     string: Option<String>, | ||||
| ) -> std::result::Result<Option<OffsetDateTime>, TakeErrorMessage<ParseOffsetDateTimeError>> { | ||||
| ) -> std::result::Result<Option<OffsetDateTime>, ParseOffsetDateTimeError> { | ||||
|     let Some(string) = string else { | ||||
|         return Ok(None) | ||||
|     }; | ||||
| @@ -186,12 +160,12 @@ fn parse_expiration_date( | ||||
|     ) { | ||||
|         PrimitiveDateTime::new(date, time!(00:00)).assume_utc() | ||||
|     } else { | ||||
|         return Err(TakeErrorMessage(ParseOffsetDateTimeError(string))); | ||||
|         return Err(ParseOffsetDateTimeError(string)); | ||||
|     }; | ||||
|     if datetime > OffsetDateTime::now_utc() { | ||||
|         Ok(Some(datetime)) | ||||
|     } else { | ||||
|         Err(TakeErrorMessage(ParseOffsetDateTimeError(string))) | ||||
|         Err(ParseOffsetDateTimeError(string)) | ||||
|     } | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -7,7 +7,7 @@ pub mod settings; | ||||
| pub mod star_or; | ||||
| pub mod tasks; | ||||
| pub mod versioning; | ||||
|  | ||||
| pub mod deserr; | ||||
| pub use milli; | ||||
| pub use milli::{heed, Index}; | ||||
| pub use serde_cs; | ||||
|   | ||||
| @@ -11,8 +11,9 @@ use milli::update::Setting; | ||||
| use milli::{Criterion, CriterionError, Index, DEFAULT_VALUES_PER_FACET}; | ||||
| use serde::{Deserialize, Serialize, Serializer}; | ||||
|  | ||||
| use crate::deserr::DeserrJsonError; | ||||
| use crate::error::deserr_codes::*; | ||||
| use crate::error::{unwrap_any, DeserrJsonError}; | ||||
| use crate::error::{unwrap_any}; | ||||
|  | ||||
| /// The maximimum number of results that the engine | ||||
| /// will be able to return in one search call. | ||||
|   | ||||
| @@ -1,13 +1,9 @@ | ||||
| use std::fmt::{Display, Formatter}; | ||||
| use std::marker::PhantomData; | ||||
| use std::ops::Deref; | ||||
| use std::str::FromStr; | ||||
| use std::{fmt, marker::PhantomData, str::FromStr}; | ||||
|  | ||||
| use deserr::{DeserializeError, DeserializeFromValue, MergeWithError, ValueKind}; | ||||
| use serde::de::Visitor; | ||||
| use serde::{Deserialize, Deserializer, Serialize, Serializer}; | ||||
| use serde::{de::Visitor, Deserialize, Deserializer, Serialize, Serializer}; | ||||
|  | ||||
| use crate::error::unwrap_any; | ||||
| use crate::{deserr::query_params::FromQueryParameter, error::unwrap_any}; | ||||
|  | ||||
| /// A type that tries to match either a star (*) or | ||||
| /// any other thing that implements `FromStr`. | ||||
| @@ -17,35 +13,6 @@ pub enum StarOr<T> { | ||||
|     Other(T), | ||||
| } | ||||
|  | ||||
| impl<E: DeserializeError, T> DeserializeFromValue<E> for StarOr<T> | ||||
| where | ||||
|     T: FromStr, | ||||
|     E: MergeWithError<T::Err>, | ||||
| { | ||||
|     fn deserialize_from_value<V: deserr::IntoValue>( | ||||
|         value: deserr::Value<V>, | ||||
|         location: deserr::ValuePointerRef, | ||||
|     ) -> Result<Self, E> { | ||||
|         match value { | ||||
|             deserr::Value::String(v) => match v.as_str() { | ||||
|                 "*" => Ok(StarOr::Star), | ||||
|                 v => match FromStr::from_str(v) { | ||||
|                     Ok(x) => Ok(StarOr::Other(x)), | ||||
|                     Err(e) => Err(unwrap_any(E::merge(None, e, location))), | ||||
|                 }, | ||||
|             }, | ||||
|             _ => Err(unwrap_any(E::error::<V>( | ||||
|                 None, | ||||
|                 deserr::ErrorKind::IncorrectValueKind { | ||||
|                     actual: value, | ||||
|                     accepted: &[ValueKind::String], | ||||
|                 }, | ||||
|                 location, | ||||
|             ))), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<T: FromStr> FromStr for StarOr<T> { | ||||
|     type Err = T::Err; | ||||
|  | ||||
| @@ -57,23 +24,11 @@ impl<T: FromStr> FromStr for StarOr<T> { | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<T: Deref<Target = str>> Deref for StarOr<T> { | ||||
|     type Target = str; | ||||
|  | ||||
|     fn deref(&self) -> &Self::Target { | ||||
| impl<T: fmt::Display> fmt::Display for StarOr<T> { | ||||
|     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||||
|         match self { | ||||
|             Self::Star => "*", | ||||
|             Self::Other(t) => t.deref(), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<T: Into<String>> From<StarOr<T>> for String { | ||||
|     fn from(s: StarOr<T>) -> Self { | ||||
|         match s { | ||||
|             StarOr::Star => "*".to_string(), | ||||
|             StarOr::Other(t) => t.into(), | ||||
|             StarOr::Star => write!(f, "*"), | ||||
|             StarOr::Other(x) => fmt::Display::fmt(x, f), | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -93,7 +48,7 @@ impl<T: PartialEq + Eq> Eq for StarOr<T> {} | ||||
| impl<'de, T, E> Deserialize<'de> for StarOr<T> | ||||
| where | ||||
|     T: FromStr<Err = E>, | ||||
|     E: Display, | ||||
|     E: fmt::Display, | ||||
| { | ||||
|     fn deserialize<D>(deserializer: D) -> Result<Self, D::Error> | ||||
|     where | ||||
| @@ -109,11 +64,11 @@ where | ||||
|         impl<'de, T, FE> Visitor<'de> for StarOrVisitor<T> | ||||
|         where | ||||
|             T: FromStr<Err = FE>, | ||||
|             FE: Display, | ||||
|             FE: fmt::Display, | ||||
|         { | ||||
|             type Value = StarOr<T>; | ||||
|  | ||||
|             fn expecting(&self, formatter: &mut Formatter) -> std::fmt::Result { | ||||
|             fn expecting(&self, formatter: &mut fmt::Formatter) -> std::fmt::Result { | ||||
|                 formatter.write_str("a string") | ||||
|             } | ||||
|  | ||||
| @@ -139,7 +94,7 @@ where | ||||
|  | ||||
| impl<T> Serialize for StarOr<T> | ||||
| where | ||||
|     T: Deref<Target = str>, | ||||
|     T: ToString, | ||||
| { | ||||
|     fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error> | ||||
|     where | ||||
| @@ -147,7 +102,222 @@ where | ||||
|     { | ||||
|         match self { | ||||
|             StarOr::Star => serializer.serialize_str("*"), | ||||
|             StarOr::Other(other) => serializer.serialize_str(other.deref()), | ||||
|             StarOr::Other(other) => serializer.serialize_str(&other.to_string()), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<T, E> DeserializeFromValue<E> for StarOr<T> | ||||
| where | ||||
|     T: FromStr, | ||||
|     E: DeserializeError + MergeWithError<T::Err>, | ||||
| { | ||||
|     fn deserialize_from_value<V: deserr::IntoValue>( | ||||
|         value: deserr::Value<V>, | ||||
|         location: deserr::ValuePointerRef, | ||||
|     ) -> Result<Self, E> { | ||||
|         match value { | ||||
|             deserr::Value::String(v) => { | ||||
|                 if v == "*" { | ||||
|                     Ok(StarOr::Star) | ||||
|                 } else { | ||||
|                     match T::from_str(&v) { | ||||
|                         Ok(parsed) => Ok(StarOr::Other(parsed)), | ||||
|                         Err(e) => Err(unwrap_any(E::merge(None, e, location))), | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             _ => Err(unwrap_any(E::error::<V>( | ||||
|                 None, | ||||
|                 deserr::ErrorKind::IncorrectValueKind { | ||||
|                     actual: value, | ||||
|                     accepted: &[ValueKind::String], | ||||
|                 }, | ||||
|                 location, | ||||
|             ))), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// A type representing the content of a query parameter that can either not exist, | ||||
| /// be equal to a star (*), or another value | ||||
| /// | ||||
| /// It is a convenient alternative to `Option<StarOr<T>>`. | ||||
| #[derive(Debug, Default, Clone, Copy)] | ||||
| pub enum OptionStarOr<T> { | ||||
|     #[default] | ||||
|     None, | ||||
|     Star, | ||||
|     Other(T), | ||||
| } | ||||
|  | ||||
| impl<T> OptionStarOr<T> { | ||||
|     pub fn is_some(&self) -> bool { | ||||
|         match self { | ||||
|             Self::None => false, | ||||
|             Self::Star => false, | ||||
|             Self::Other(_) => true, | ||||
|         } | ||||
|     } | ||||
|     pub fn merge_star_and_none(self) -> Option<T> { | ||||
|         match self { | ||||
|             Self::None | Self::Star => None, | ||||
|             Self::Other(x) => Some(x), | ||||
|         } | ||||
|     } | ||||
|     pub fn try_map<U, E, F: Fn(T) -> Result<U, E>>(self, map_f: F) -> Result<OptionStarOr<U>, E> { | ||||
|         match self { | ||||
|             OptionStarOr::None => Ok(OptionStarOr::None), | ||||
|             OptionStarOr::Star => Ok(OptionStarOr::Star), | ||||
|             OptionStarOr::Other(x) => map_f(x).map(OptionStarOr::Other), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<T> FromQueryParameter for OptionStarOr<T> | ||||
| where | ||||
|     T: FromQueryParameter, | ||||
| { | ||||
|     type Err = T::Err; | ||||
|     fn from_query_param(p: &str) -> Result<Self, Self::Err> { | ||||
|         match p { | ||||
|             "*" => Ok(OptionStarOr::Star), | ||||
|             s => T::from_query_param(s).map(OptionStarOr::Other), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<T, E> DeserializeFromValue<E> for OptionStarOr<T> | ||||
| where | ||||
|     E: DeserializeError + MergeWithError<T::Err>, | ||||
|     T: FromQueryParameter, | ||||
| { | ||||
|     fn deserialize_from_value<V: deserr::IntoValue>( | ||||
|         value: deserr::Value<V>, | ||||
|         location: deserr::ValuePointerRef, | ||||
|     ) -> Result<Self, E> { | ||||
|         match value { | ||||
|             deserr::Value::String(s) => match s.as_str() { | ||||
|                 "*" => Ok(OptionStarOr::Star), | ||||
|                 s => match T::from_query_param(s) { | ||||
|                     Ok(x) => Ok(OptionStarOr::Other(x)), | ||||
|                     Err(e) => Err(unwrap_any(E::merge(None, e, location))), | ||||
|                 }, | ||||
|             }, | ||||
|             _ => Err(unwrap_any(E::error::<V>( | ||||
|                 None, | ||||
|                 deserr::ErrorKind::IncorrectValueKind { | ||||
|                     actual: value, | ||||
|                     accepted: &[ValueKind::String], | ||||
|                 }, | ||||
|                 location, | ||||
|             ))), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// A type representing the content of a query parameter that can either not exist, be equal to a star (*), or represent a list of other values | ||||
| #[derive(Debug, Default, Clone)] | ||||
| pub enum OptionStarOrList<T> { | ||||
|     #[default] | ||||
|     None, | ||||
|     Star, | ||||
|     List(Vec<T>), | ||||
| } | ||||
|  | ||||
| impl<T> OptionStarOrList<T> { | ||||
|     pub fn is_some(&self) -> bool { | ||||
|         match self { | ||||
|             Self::None => false, | ||||
|             Self::Star => false, | ||||
|             Self::List(_) => true, | ||||
|         } | ||||
|     } | ||||
|     pub fn map<U, F: Fn(T) -> U>(self, map_f: F) -> OptionStarOrList<U> { | ||||
|         match self { | ||||
|             Self::None => OptionStarOrList::None, | ||||
|             Self::Star => OptionStarOrList::Star, | ||||
|             Self::List(xs) => OptionStarOrList::List(xs.into_iter().map(map_f).collect()), | ||||
|         } | ||||
|     } | ||||
|     pub fn try_map<U, E, F: Fn(T) -> Result<U, E>>( | ||||
|         self, | ||||
|         map_f: F, | ||||
|     ) -> Result<OptionStarOrList<U>, E> { | ||||
|         match self { | ||||
|             Self::None => Ok(OptionStarOrList::None), | ||||
|             Self::Star => Ok(OptionStarOrList::Star), | ||||
|             Self::List(xs) => { | ||||
|                 xs.into_iter().map(map_f).collect::<Result<Vec<_>, _>>().map(OptionStarOrList::List) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     pub fn merge_star_and_none(self) -> Option<Vec<T>> { | ||||
|         match self { | ||||
|             Self::None | Self::Star => None, | ||||
|             Self::List(xs) => Some(xs), | ||||
|         } | ||||
|     } | ||||
|     pub fn push(&mut self, el: T) { | ||||
|         match self { | ||||
|             Self::None => *self = Self::List(vec![el]), | ||||
|             Self::Star => (), | ||||
|             Self::List(xs) => xs.push(el), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<T, E> DeserializeFromValue<E> for OptionStarOrList<T> | ||||
| where | ||||
|     E: DeserializeError + MergeWithError<T::Err>, | ||||
|     T: FromQueryParameter, | ||||
| { | ||||
|     fn deserialize_from_value<V: deserr::IntoValue>( | ||||
|         value: deserr::Value<V>, | ||||
|         location: deserr::ValuePointerRef, | ||||
|     ) -> Result<Self, E> { | ||||
|         match value { | ||||
|             deserr::Value::String(s) => { | ||||
|                 let mut error = None; | ||||
|                 let mut is_star = false; | ||||
|                 // CS::<String>::from_str is infaillible | ||||
|                 let cs = serde_cs::vec::CS::<String>::from_str(&s).unwrap(); | ||||
|                 let len_cs = cs.0.len(); | ||||
|                 let mut els = vec![]; | ||||
|                 for (i, el_str) in cs.into_iter().enumerate() { | ||||
|                     if el_str == "*" { | ||||
|                         is_star = true; | ||||
|                     } else { | ||||
|                         match T::from_query_param(&el_str) { | ||||
|                             Ok(el) => { | ||||
|                                 els.push(el); | ||||
|                             } | ||||
|                             Err(e) => { | ||||
|                                 let location = | ||||
|                                     if len_cs > 1 { location.push_index(i) } else { location }; | ||||
|                                 error = Some(E::merge(error, e, location)?); | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|                 if let Some(error) = error { | ||||
|                     return Err(error); | ||||
|                 } | ||||
|  | ||||
|                 if is_star { | ||||
|                     Ok(OptionStarOrList::Star) | ||||
|                 } else { | ||||
|                     Ok(OptionStarOrList::List(els)) | ||||
|                 } | ||||
|             } | ||||
|             _ => Err(unwrap_any(E::error::<V>( | ||||
|                 None, | ||||
|                 deserr::ErrorKind::IncorrectValueKind { | ||||
|                     actual: value, | ||||
|                     accepted: &[ValueKind::String], | ||||
|                 }, | ||||
|                 location, | ||||
|             ))), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| use core::fmt; | ||||
| use std::collections::HashSet; | ||||
| use std::fmt::{Display, Write}; | ||||
| use std::str::FromStr; | ||||
| @@ -9,7 +10,7 @@ use serde::{Deserialize, Serialize, Serializer}; | ||||
| use time::{Duration, OffsetDateTime}; | ||||
| use uuid::Uuid; | ||||
|  | ||||
| use crate::error::{Code, ResponseError}; | ||||
| use crate::error::ResponseError; | ||||
| use crate::keys::Key; | ||||
| use crate::settings::{Settings, Unchecked}; | ||||
| use crate::InstanceUid; | ||||
| @@ -332,7 +333,7 @@ impl Display for Status { | ||||
| } | ||||
|  | ||||
| impl FromStr for Status { | ||||
|     type Err = ResponseError; | ||||
|     type Err = ParseTaskStatusError; | ||||
|  | ||||
|     fn from_str(status: &str) -> Result<Self, Self::Err> { | ||||
|         if status.eq_ignore_ascii_case("enqueued") { | ||||
| @@ -346,20 +347,27 @@ impl FromStr for Status { | ||||
|         } else if status.eq_ignore_ascii_case("canceled") { | ||||
|             Ok(Status::Canceled) | ||||
|         } else { | ||||
|             Err(ResponseError::from_msg( | ||||
|                 format!( | ||||
|             Err(ParseTaskStatusError(status.to_owned())) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Debug)] | ||||
| pub struct ParseTaskStatusError(pub String); | ||||
| impl fmt::Display for ParseTaskStatusError { | ||||
|     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||||
|         write!( | ||||
|             f, | ||||
|             "`{}` is not a valid task status. Available statuses are {}.", | ||||
|                     status, | ||||
|             self.0, | ||||
|             enum_iterator::all::<Status>() | ||||
|                 .map(|s| format!("`{s}`")) | ||||
|                 .collect::<Vec<String>>() | ||||
|                 .join(", ") | ||||
|                 ), | ||||
|                 Code::BadRequest, | ||||
|             )) | ||||
|         } | ||||
|         ) | ||||
|     } | ||||
| } | ||||
| impl std::error::Error for ParseTaskStatusError {} | ||||
|  | ||||
| #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, Serialize, Deserialize, Sequence)] | ||||
| #[serde(rename_all = "camelCase")] | ||||
| @@ -412,7 +420,7 @@ impl Display for Kind { | ||||
|     } | ||||
| } | ||||
| impl FromStr for Kind { | ||||
|     type Err = ResponseError; | ||||
|     type Err = ParseTaskKindError; | ||||
|  | ||||
|     fn from_str(kind: &str) -> Result<Self, Self::Err> { | ||||
|         if kind.eq_ignore_ascii_case("indexCreation") { | ||||
| @@ -438,10 +446,19 @@ impl FromStr for Kind { | ||||
|         } else if kind.eq_ignore_ascii_case("snapshotCreation") { | ||||
|             Ok(Kind::SnapshotCreation) | ||||
|         } else { | ||||
|             Err(ResponseError::from_msg( | ||||
|                 format!( | ||||
|             Err(ParseTaskKindError(kind.to_owned())) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Debug)] | ||||
| pub struct ParseTaskKindError(pub String); | ||||
| impl fmt::Display for ParseTaskKindError { | ||||
|     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||||
|         write!( | ||||
|             f, | ||||
|             "`{}` is not a valid task type. Available types are {}.", | ||||
|                     kind, | ||||
|             self.0, | ||||
|             enum_iterator::all::<Kind>() | ||||
|                 .map(|k| format!( | ||||
|                     "`{}`", | ||||
| @@ -450,12 +467,10 @@ impl FromStr for Kind { | ||||
|                 )) | ||||
|                 .collect::<Vec<String>>() | ||||
|                 .join(", ") | ||||
|                 ), | ||||
|                 Code::BadRequest, | ||||
|             )) | ||||
|         } | ||||
|         ) | ||||
|     } | ||||
| } | ||||
| impl std::error::Error for ParseTaskKindError {} | ||||
|  | ||||
| #[derive(Debug, PartialEq, Eq, Clone, Serialize, Deserialize)] | ||||
| pub enum Details { | ||||
|   | ||||
| @@ -19,7 +19,7 @@ byte-unit = { version = "4.0.14", default-features = false, features = ["std", " | ||||
| bytes = "1.2.1" | ||||
| clap = { version = "4.0.9", features = ["derive", "env"] } | ||||
| crossbeam-channel = "0.5.6" | ||||
| deserr = { path = "/Users/meilisearch/Documents/deserr" } | ||||
| deserr = "0.1.4" | ||||
| dump = { path = "../dump" } | ||||
| either = "1.8.0" | ||||
| env_logger = "0.9.1" | ||||
|   | ||||
| @@ -4,14 +4,15 @@ use actix_web::{web, HttpRequest, HttpResponse}; | ||||
| use deserr::DeserializeFromValue; | ||||
| use meilisearch_auth::error::AuthControllerError; | ||||
| use meilisearch_auth::AuthController; | ||||
| use meilisearch_types::error::{deserr_codes::*, DeserrQueryParamError}; | ||||
| use meilisearch_types::error::{Code, DeserrJsonError, ResponseError, TakeErrorMessage}; | ||||
| use meilisearch_types::deserr::query_params::Param; | ||||
| use meilisearch_types::deserr::{DeserrJsonError, DeserrQueryParamError}; | ||||
| use meilisearch_types::error::deserr_codes::*; | ||||
| use meilisearch_types::error::{Code, ResponseError}; | ||||
| use meilisearch_types::keys::{Action, CreateApiKey, Key, PatchApiKey}; | ||||
| use serde::{Deserialize, Serialize}; | ||||
| use time::OffsetDateTime; | ||||
| use uuid::Uuid; | ||||
|  | ||||
| use super::indexes::search::parse_usize_take_error_message; | ||||
| use super::PAGINATION_DEFAULT_LIMIT; | ||||
| use crate::extractors::authentication::policies::*; | ||||
| use crate::extractors::authentication::GuardedData; | ||||
| @@ -50,20 +51,17 @@ pub async fn create_api_key( | ||||
|     Ok(HttpResponse::Created().json(res)) | ||||
| } | ||||
|  | ||||
| #[derive(DeserializeFromValue, Deserialize, Debug, Clone, Copy)] | ||||
| #[derive(DeserializeFromValue, Debug, Clone, Copy)] | ||||
| #[deserr(error = DeserrQueryParamError, rename_all = camelCase, deny_unknown_fields)] | ||||
| #[serde(rename_all = "camelCase", deny_unknown_fields)] | ||||
| pub struct ListApiKeys { | ||||
|     #[serde(default)] | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidApiKeyOffset>, from(&String) = parse_usize_take_error_message -> TakeErrorMessage<std::num::ParseIntError>)] | ||||
|     pub offset: usize, | ||||
|     #[serde(default = "PAGINATION_DEFAULT_LIMIT")] | ||||
|     #[deserr(default = PAGINATION_DEFAULT_LIMIT(), error = DeserrQueryParamError<InvalidApiKeyLimit>, from(&String) = parse_usize_take_error_message -> TakeErrorMessage<std::num::ParseIntError>)] | ||||
|     pub limit: usize, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidApiKeyOffset>)] | ||||
|     pub offset: Param<usize>, | ||||
|     #[deserr(default = Param(PAGINATION_DEFAULT_LIMIT), error = DeserrQueryParamError<InvalidApiKeyLimit>)] | ||||
|     pub limit: Param<usize>, | ||||
| } | ||||
| impl ListApiKeys { | ||||
|     fn as_pagination(self) -> Pagination { | ||||
|         Pagination { offset: self.offset, limit: self.limit } | ||||
|         Pagination { offset: self.offset.0, limit: self.limit.0 } | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -172,7 +170,7 @@ impl KeyView { | ||||
|             key: generated_key, | ||||
|             uid: key.uid, | ||||
|             actions: key.actions, | ||||
|             indexes: key.indexes.into_iter().map(String::from).collect(), | ||||
|             indexes: key.indexes.into_iter().map(|x| x.to_string()).collect(), | ||||
|             expires_at: key.expires_at, | ||||
|             created_at: key.created_at, | ||||
|             updated_at: key.updated_at, | ||||
|   | ||||
| @@ -1,5 +1,4 @@ | ||||
| use std::io::ErrorKind; | ||||
| use std::num::ParseIntError; | ||||
|  | ||||
| use actix_web::http::header::CONTENT_TYPE; | ||||
| use actix_web::web::Data; | ||||
| @@ -9,14 +8,15 @@ use deserr::DeserializeFromValue; | ||||
| use futures::StreamExt; | ||||
| use index_scheduler::IndexScheduler; | ||||
| use log::debug; | ||||
| use meilisearch_types::deserr::query_params::Param; | ||||
| use meilisearch_types::deserr::{DeserrJsonError, DeserrQueryParamError}; | ||||
| use meilisearch_types::document_formats::{read_csv, read_json, read_ndjson, PayloadType}; | ||||
| use meilisearch_types::error::{deserr_codes::*, fold_star_or, DeserrQueryParamError}; | ||||
| use meilisearch_types::error::{DeserrJsonError, ResponseError, TakeErrorMessage}; | ||||
| use meilisearch_types::error::deserr_codes::*; | ||||
| use meilisearch_types::error::ResponseError; | ||||
| use meilisearch_types::heed::RoTxn; | ||||
| use meilisearch_types::index_uid::IndexUid; | ||||
| use meilisearch_types::milli::update::IndexDocumentsMethod; | ||||
| use meilisearch_types::serde_cs::vec::CS; | ||||
| use meilisearch_types::star_or::StarOr; | ||||
| use meilisearch_types::star_or::OptionStarOrList; | ||||
| use meilisearch_types::tasks::KindWithContent; | ||||
| use meilisearch_types::{milli, Document, Index}; | ||||
| use mime::Mime; | ||||
| @@ -27,7 +27,6 @@ use tempfile::tempfile; | ||||
| use tokio::fs::File; | ||||
| use tokio::io::{AsyncSeekExt, AsyncWriteExt, BufWriter}; | ||||
|  | ||||
| use super::search::parse_usize_take_error_message; | ||||
| use crate::analytics::{Analytics, DocumentDeletionKind}; | ||||
| use crate::error::MeilisearchHttpError; | ||||
| use crate::error::PayloadError::ReceivePayload; | ||||
| @@ -36,7 +35,7 @@ use crate::extractors::authentication::GuardedData; | ||||
| use crate::extractors::payload::Payload; | ||||
| use crate::extractors::query_parameters::QueryParameter; | ||||
| use crate::extractors::sequential_extractor::SeqHandler; | ||||
| use crate::routes::{PaginationView, SummarizedTaskView}; | ||||
| use crate::routes::{PaginationView, SummarizedTaskView, PAGINATION_DEFAULT_LIMIT}; | ||||
|  | ||||
| static ACCEPTED_CONTENT_TYPE: Lazy<Vec<String>> = Lazy::new(|| { | ||||
|     vec!["application/json".to_string(), "application/x-ndjson".to_string(), "text/csv".to_string()] | ||||
| @@ -81,12 +80,11 @@ pub fn configure(cfg: &mut web::ServiceConfig) { | ||||
|     ); | ||||
| } | ||||
|  | ||||
| #[derive(Deserialize, Debug, DeserializeFromValue)] | ||||
| #[derive(Debug, DeserializeFromValue)] | ||||
| #[deserr(error = DeserrQueryParamError, rename_all = camelCase, deny_unknown_fields)] | ||||
| pub struct GetDocument { | ||||
|     // TODO: strongly typed argument here | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidDocumentFields>)] | ||||
|     fields: Option<CS<StarOr<String>>>, | ||||
|     fields: OptionStarOrList<String>, | ||||
| } | ||||
|  | ||||
| pub async fn get_document( | ||||
| @@ -95,7 +93,7 @@ pub async fn get_document( | ||||
|     params: QueryParameter<GetDocument, DeserrQueryParamError>, | ||||
| ) -> Result<HttpResponse, ResponseError> { | ||||
|     let GetDocument { fields } = params.into_inner(); | ||||
|     let attributes_to_retrieve = fields.and_then(fold_star_or); | ||||
|     let attributes_to_retrieve = fields.merge_star_and_none(); | ||||
|  | ||||
|     let index = index_scheduler.index(&path.index_uid)?; | ||||
|     let document = retrieve_document(&index, &path.document_id, attributes_to_retrieve)?; | ||||
| @@ -119,15 +117,15 @@ pub async fn delete_document( | ||||
|     Ok(HttpResponse::Accepted().json(task)) | ||||
| } | ||||
|  | ||||
| #[derive(Deserialize, Debug, DeserializeFromValue)] | ||||
| #[derive(Debug, DeserializeFromValue)] | ||||
| #[deserr(error = DeserrQueryParamError, rename_all = camelCase, deny_unknown_fields)] | ||||
| pub struct BrowseQuery { | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidDocumentFields>, from(&String) = parse_usize_take_error_message -> TakeErrorMessage<ParseIntError>)] | ||||
|     offset: usize, | ||||
|     #[deserr(default = crate::routes::PAGINATION_DEFAULT_LIMIT(), error = DeserrQueryParamError<InvalidDocumentLimit>, from(&String) = parse_usize_take_error_message -> TakeErrorMessage<ParseIntError>)] | ||||
|     limit: usize, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidDocumentFields>)] | ||||
|     offset: Param<usize>, | ||||
|     #[deserr(default = Param(PAGINATION_DEFAULT_LIMIT), error = DeserrQueryParamError<InvalidDocumentLimit>)] | ||||
|     limit: Param<usize>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidDocumentLimit>)] | ||||
|     fields: Option<CS<StarOr<String>>>, | ||||
|     fields: OptionStarOrList<String>, | ||||
| } | ||||
|  | ||||
| pub async fn get_all_documents( | ||||
| @@ -137,12 +135,12 @@ pub async fn get_all_documents( | ||||
| ) -> Result<HttpResponse, ResponseError> { | ||||
|     debug!("called with params: {:?}", params); | ||||
|     let BrowseQuery { limit, offset, fields } = params.into_inner(); | ||||
|     let attributes_to_retrieve = fields.and_then(fold_star_or); | ||||
|     let attributes_to_retrieve = fields.merge_star_and_none(); | ||||
|  | ||||
|     let index = index_scheduler.index(&index_uid)?; | ||||
|     let (total, documents) = retrieve_documents(&index, offset, limit, attributes_to_retrieve)?; | ||||
|     let (total, documents) = retrieve_documents(&index, offset.0, limit.0, attributes_to_retrieve)?; | ||||
|  | ||||
|     let ret = PaginationView::new(offset, limit, total as usize, documents); | ||||
|     let ret = PaginationView::new(offset.0, limit.0, total as usize, documents); | ||||
|  | ||||
|     debug!("returns: {:?}", ret); | ||||
|     Ok(HttpResponse::Ok().json(ret)) | ||||
|   | ||||
| @@ -5,8 +5,10 @@ use actix_web::{web, HttpRequest, HttpResponse}; | ||||
| use deserr::{DeserializeError, DeserializeFromValue, ValuePointerRef}; | ||||
| use index_scheduler::IndexScheduler; | ||||
| use log::debug; | ||||
| use meilisearch_types::error::{deserr_codes::*, unwrap_any, Code, DeserrQueryParamError}; | ||||
| use meilisearch_types::error::{DeserrJsonError, ResponseError, TakeErrorMessage}; | ||||
| use meilisearch_types::deserr::query_params::Param; | ||||
| use meilisearch_types::deserr::{DeserrJsonError, DeserrQueryParamError}; | ||||
| use meilisearch_types::error::ResponseError; | ||||
| use meilisearch_types::error::{deserr_codes::*, unwrap_any, Code}; | ||||
| use meilisearch_types::index_uid::IndexUid; | ||||
| use meilisearch_types::milli::{self, FieldDistribution, Index}; | ||||
| use meilisearch_types::tasks::KindWithContent; | ||||
| @@ -14,7 +16,6 @@ use serde::{Deserialize, Serialize}; | ||||
| use serde_json::json; | ||||
| use time::OffsetDateTime; | ||||
|  | ||||
| use self::search::parse_usize_take_error_message; | ||||
| use super::{Pagination, SummarizedTaskView, PAGINATION_DEFAULT_LIMIT}; | ||||
| use crate::analytics::Analytics; | ||||
| use crate::extractors::authentication::policies::*; | ||||
| @@ -71,20 +72,17 @@ impl IndexView { | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(DeserializeFromValue, Deserialize, Debug, Clone, Copy)] | ||||
| #[derive(DeserializeFromValue, Debug, Clone, Copy)] | ||||
| #[deserr(error = DeserrQueryParamError, rename_all = camelCase, deny_unknown_fields)] | ||||
| #[serde(rename_all = "camelCase", deny_unknown_fields)] | ||||
| pub struct ListIndexes { | ||||
|     #[serde(default)] | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidIndexOffset>, from(&String) = parse_usize_take_error_message -> TakeErrorMessage<std::num::ParseIntError>)] | ||||
|     pub offset: usize, | ||||
|     #[serde(default = "PAGINATION_DEFAULT_LIMIT")] | ||||
|     #[deserr(default = PAGINATION_DEFAULT_LIMIT(), error = DeserrQueryParamError<InvalidIndexLimit>, from(&String) = parse_usize_take_error_message -> TakeErrorMessage<std::num::ParseIntError>)] | ||||
|     pub limit: usize, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidIndexOffset>)] | ||||
|     pub offset: Param<usize>, | ||||
|     #[deserr(default = Param(PAGINATION_DEFAULT_LIMIT), error = DeserrQueryParamError<InvalidIndexLimit>)] | ||||
|     pub limit: Param<usize>, | ||||
| } | ||||
| impl ListIndexes { | ||||
|     fn as_pagination(self) -> Pagination { | ||||
|         Pagination { offset: self.offset, limit: self.limit } | ||||
|         Pagination { offset: self.offset.0, limit: self.limit.0 } | ||||
|     } | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,15 +1,12 @@ | ||||
| use std::str::FromStr; | ||||
|  | ||||
| use actix_web::web::Data; | ||||
| use actix_web::{web, HttpRequest, HttpResponse}; | ||||
| use index_scheduler::IndexScheduler; | ||||
| use log::debug; | ||||
| use meilisearch_auth::IndexSearchRules; | ||||
| use meilisearch_types::error::{ | ||||
|     deserr_codes::*, parse_option_usize_query_param, parse_usize_query_param, | ||||
|     DeserrQueryParamError, DetailedParseIntError, | ||||
| }; | ||||
| use meilisearch_types::error::{DeserrJsonError, ResponseError, TakeErrorMessage}; | ||||
| use meilisearch_types::deserr::{DeserrQueryParamError, DeserrJsonError}; | ||||
| use meilisearch_types::deserr::query_params::Param; | ||||
| use meilisearch_types::error::deserr_codes::*; | ||||
| use meilisearch_types::error::ResponseError; | ||||
| use meilisearch_types::serde_cs::vec::CS; | ||||
| use serde_json::Value; | ||||
|  | ||||
| @@ -33,45 +30,33 @@ pub fn configure(cfg: &mut web::ServiceConfig) { | ||||
|     ); | ||||
| } | ||||
|  | ||||
| pub fn parse_usize_take_error_message( | ||||
|     s: &str, | ||||
| ) -> Result<usize, TakeErrorMessage<std::num::ParseIntError>> { | ||||
|     usize::from_str(s).map_err(TakeErrorMessage) | ||||
| } | ||||
|  | ||||
| pub fn parse_bool_take_error_message( | ||||
|     s: &str, | ||||
| ) -> Result<bool, TakeErrorMessage<std::str::ParseBoolError>> { | ||||
|     s.parse().map_err(TakeErrorMessage) | ||||
| } | ||||
|  | ||||
| #[derive(Debug, deserr::DeserializeFromValue)] | ||||
| #[deserr(error = DeserrQueryParamError, rename_all = camelCase, deny_unknown_fields)] | ||||
| pub struct SearchQueryGet { | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidSearchQ>)] | ||||
|     q: Option<String>, | ||||
|     #[deserr(default = DEFAULT_SEARCH_OFFSET(), error = DeserrQueryParamError<InvalidSearchOffset>, from(String) = parse_usize_query_param -> TakeErrorMessage<DetailedParseIntError>)] | ||||
|     offset: usize, | ||||
|     #[deserr(default = DEFAULT_SEARCH_LIMIT(), error = DeserrQueryParamError<InvalidSearchLimit>, from(String) = parse_usize_query_param -> TakeErrorMessage<DetailedParseIntError>)] | ||||
|     limit: usize, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidSearchPage>, from(Option<String>) = parse_option_usize_query_param -> TakeErrorMessage<std::num::ParseIntError>)] | ||||
|     page: Option<usize>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidSearchHitsPerPage>, from(Option<String>) = parse_option_usize_query_param -> TakeErrorMessage<std::num::ParseIntError>)] | ||||
|     hits_per_page: Option<usize>, | ||||
|     #[deserr(default = Param(DEFAULT_SEARCH_OFFSET()), error = DeserrQueryParamError<InvalidSearchOffset>)] | ||||
|     offset: Param<usize>, | ||||
|     #[deserr(default = Param(DEFAULT_SEARCH_LIMIT()), error = DeserrQueryParamError<InvalidSearchLimit>)] | ||||
|     limit: Param<usize>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidSearchPage>)] | ||||
|     page: Option<Param<usize>>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidSearchHitsPerPage>)] | ||||
|     hits_per_page: Option<Param<usize>>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidSearchAttributesToRetrieve>)] | ||||
|     attributes_to_retrieve: Option<CS<String>>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidSearchAttributesToCrop>)] | ||||
|     attributes_to_crop: Option<CS<String>>, | ||||
|     #[deserr(default = DEFAULT_CROP_LENGTH(), error = DeserrQueryParamError<InvalidSearchCropLength>, from(String) = parse_usize_query_param -> TakeErrorMessage<DetailedParseIntError>)] | ||||
|     crop_length: usize, | ||||
|     #[deserr(default = Param(DEFAULT_CROP_LENGTH()), error = DeserrQueryParamError<InvalidSearchCropLength>)] | ||||
|     crop_length: Param<usize>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidSearchAttributesToHighlight>)] | ||||
|     attributes_to_highlight: Option<CS<String>>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidSearchFilter>)] | ||||
|     filter: Option<String>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidSearchSort>)] | ||||
|     sort: Option<String>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidSearchShowMatchesPosition>, from(&String) = parse_bool_take_error_message -> TakeErrorMessage<std::str::ParseBoolError>)] | ||||
|     show_matches_position: bool, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidSearchShowMatchesPosition>)] | ||||
|     show_matches_position: Param<bool>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidSearchFacets>)] | ||||
|     facets: Option<CS<String>>, | ||||
|     #[deserr( default = DEFAULT_HIGHLIGHT_PRE_TAG(), error = DeserrQueryParamError<InvalidSearchHighlightPreTag>)] | ||||
| @@ -96,17 +81,17 @@ impl From<SearchQueryGet> for SearchQuery { | ||||
|  | ||||
|         Self { | ||||
|             q: other.q, | ||||
|             offset: other.offset, | ||||
|             limit: other.limit, | ||||
|             page: other.page, | ||||
|             hits_per_page: other.hits_per_page, | ||||
|             offset: other.offset.0, | ||||
|             limit: other.limit.0, | ||||
|             page: other.page.as_deref().copied(), | ||||
|             hits_per_page: other.hits_per_page.as_deref().copied(), | ||||
|             attributes_to_retrieve: other.attributes_to_retrieve.map(|o| o.into_iter().collect()), | ||||
|             attributes_to_crop: other.attributes_to_crop.map(|o| o.into_iter().collect()), | ||||
|             crop_length: other.crop_length, | ||||
|             crop_length: other.crop_length.0, | ||||
|             attributes_to_highlight: other.attributes_to_highlight.map(|o| o.into_iter().collect()), | ||||
|             filter, | ||||
|             sort: other.sort.map(|attr| fix_sort_query_parameters(&attr)), | ||||
|             show_matches_position: other.show_matches_position, | ||||
|             show_matches_position: other.show_matches_position.0, | ||||
|             facets: other.facets.map(|o| o.into_iter().collect()), | ||||
|             highlight_pre_tag: other.highlight_pre_tag, | ||||
|             highlight_post_tag: other.highlight_post_tag, | ||||
|   | ||||
| @@ -2,7 +2,8 @@ use actix_web::web::Data; | ||||
| use actix_web::{web, HttpRequest, HttpResponse}; | ||||
| use index_scheduler::IndexScheduler; | ||||
| use log::debug; | ||||
| use meilisearch_types::error::{DeserrJsonError, ResponseError}; | ||||
| use meilisearch_types::deserr::DeserrJsonError; | ||||
| use meilisearch_types::error::ResponseError; | ||||
| use meilisearch_types::index_uid::IndexUid; | ||||
| use meilisearch_types::settings::{settings, RankingRuleView, Settings, Unchecked}; | ||||
| use meilisearch_types::tasks::KindWithContent; | ||||
| @@ -130,7 +131,7 @@ make_setting_route!( | ||||
|     "/filterable-attributes", | ||||
|     put, | ||||
|     std::collections::BTreeSet<String>, | ||||
|     meilisearch_types::error::DeserrJsonError< | ||||
|     meilisearch_types::deserr::DeserrJsonError< | ||||
|         meilisearch_types::error::deserr_codes::InvalidSettingsFilterableAttributes, | ||||
|     >, | ||||
|     filterable_attributes, | ||||
| @@ -156,7 +157,7 @@ make_setting_route!( | ||||
|     "/sortable-attributes", | ||||
|     put, | ||||
|     std::collections::BTreeSet<String>, | ||||
|     meilisearch_types::error::DeserrJsonError< | ||||
|     meilisearch_types::deserr::DeserrJsonError< | ||||
|         meilisearch_types::error::deserr_codes::InvalidSettingsSortableAttributes, | ||||
|     >, | ||||
|     sortable_attributes, | ||||
| @@ -182,7 +183,7 @@ make_setting_route!( | ||||
|     "/displayed-attributes", | ||||
|     put, | ||||
|     Vec<String>, | ||||
|     meilisearch_types::error::DeserrJsonError< | ||||
|     meilisearch_types::deserr::DeserrJsonError< | ||||
|         meilisearch_types::error::deserr_codes::InvalidSettingsDisplayedAttributes, | ||||
|     >, | ||||
|     displayed_attributes, | ||||
| @@ -208,7 +209,7 @@ make_setting_route!( | ||||
|     "/typo-tolerance", | ||||
|     patch, | ||||
|     meilisearch_types::settings::TypoSettings, | ||||
|     meilisearch_types::error::DeserrJsonError< | ||||
|     meilisearch_types::deserr::DeserrJsonError< | ||||
|         meilisearch_types::error::deserr_codes::InvalidSettingsTypoTolerance, | ||||
|     >, | ||||
|     typo_tolerance, | ||||
| @@ -253,7 +254,7 @@ make_setting_route!( | ||||
|     "/searchable-attributes", | ||||
|     put, | ||||
|     Vec<String>, | ||||
|     meilisearch_types::error::DeserrJsonError< | ||||
|     meilisearch_types::deserr::DeserrJsonError< | ||||
|         meilisearch_types::error::deserr_codes::InvalidSettingsSearchableAttributes, | ||||
|     >, | ||||
|     searchable_attributes, | ||||
| @@ -279,7 +280,7 @@ make_setting_route!( | ||||
|     "/stop-words", | ||||
|     put, | ||||
|     std::collections::BTreeSet<String>, | ||||
|     meilisearch_types::error::DeserrJsonError< | ||||
|     meilisearch_types::deserr::DeserrJsonError< | ||||
|         meilisearch_types::error::deserr_codes::InvalidSettingsStopWords, | ||||
|     >, | ||||
|     stop_words, | ||||
| @@ -304,7 +305,7 @@ make_setting_route!( | ||||
|     "/synonyms", | ||||
|     put, | ||||
|     std::collections::BTreeMap<String, Vec<String>>, | ||||
|     meilisearch_types::error::DeserrJsonError< | ||||
|     meilisearch_types::deserr::DeserrJsonError< | ||||
|         meilisearch_types::error::deserr_codes::InvalidSettingsSynonyms, | ||||
|     >, | ||||
|     synonyms, | ||||
| @@ -329,7 +330,7 @@ make_setting_route!( | ||||
|     "/distinct-attribute", | ||||
|     put, | ||||
|     String, | ||||
|     meilisearch_types::error::DeserrJsonError< | ||||
|     meilisearch_types::deserr::DeserrJsonError< | ||||
|         meilisearch_types::error::deserr_codes::InvalidSettingsDistinctAttribute, | ||||
|     >, | ||||
|     distinct_attribute, | ||||
| @@ -353,7 +354,7 @@ make_setting_route!( | ||||
|     "/ranking-rules", | ||||
|     put, | ||||
|     Vec<meilisearch_types::settings::RankingRuleView>, | ||||
|     meilisearch_types::error::DeserrJsonError< | ||||
|     meilisearch_types::deserr::DeserrJsonError< | ||||
|         meilisearch_types::error::deserr_codes::InvalidSettingsRankingRules, | ||||
|     >, | ||||
|     ranking_rules, | ||||
| @@ -384,7 +385,7 @@ make_setting_route!( | ||||
|     "/faceting", | ||||
|     patch, | ||||
|     meilisearch_types::settings::FacetingSettings, | ||||
|     meilisearch_types::error::DeserrJsonError< | ||||
|     meilisearch_types::deserr::DeserrJsonError< | ||||
|         meilisearch_types::error::deserr_codes::InvalidSettingsFaceting, | ||||
|     >, | ||||
|     faceting, | ||||
| @@ -409,7 +410,7 @@ make_setting_route!( | ||||
|     "/pagination", | ||||
|     patch, | ||||
|     meilisearch_types::settings::PaginationSettings, | ||||
|     meilisearch_types::error::DeserrJsonError< | ||||
|     meilisearch_types::deserr::DeserrJsonError< | ||||
|         meilisearch_types::error::deserr_codes::InvalidSettingsPagination, | ||||
|     >, | ||||
|     pagination, | ||||
|   | ||||
| @@ -41,7 +41,7 @@ where | ||||
|     Ok(Some(input.parse()?)) | ||||
| } | ||||
|  | ||||
| const PAGINATION_DEFAULT_LIMIT: fn() -> usize = || 20; | ||||
| const PAGINATION_DEFAULT_LIMIT: usize = 20; | ||||
|  | ||||
| #[derive(Debug, Serialize)] | ||||
| #[serde(rename_all = "camelCase")] | ||||
|   | ||||
| @@ -2,8 +2,9 @@ use actix_web::web::Data; | ||||
| use actix_web::{web, HttpRequest, HttpResponse}; | ||||
| use deserr::DeserializeFromValue; | ||||
| use index_scheduler::IndexScheduler; | ||||
| use meilisearch_types::deserr::DeserrJsonError; | ||||
| use meilisearch_types::error::deserr_codes::InvalidSwapIndexes; | ||||
| use meilisearch_types::error::{DeserrJsonError, ResponseError}; | ||||
| use meilisearch_types::error::ResponseError; | ||||
| use meilisearch_types::tasks::{IndexSwap, KindWithContent}; | ||||
| use serde_json::json; | ||||
|  | ||||
| @@ -22,7 +23,7 @@ pub fn configure(cfg: &mut web::ServiceConfig) { | ||||
| #[derive(DeserializeFromValue, Debug, Clone, PartialEq, Eq)] | ||||
| #[deserr(error = DeserrJsonError, rename_all = camelCase, deny_unknown_fields)] | ||||
| pub struct SwapIndexesPayload { | ||||
|     #[deserr(error = DeserrJsonError<InvalidSwapIndexes>, missing_field_error = DeserrJsonError::missing_swap_indexes_indexes)] | ||||
|     #[deserr(error = DeserrJsonError<InvalidSwapIndexes>, missing_field_error = DeserrJsonError::missing_swap_indexes)] | ||||
|     indexes: Vec<String>, | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -2,21 +2,17 @@ use actix_web::web::Data; | ||||
| use actix_web::{web, HttpRequest, HttpResponse}; | ||||
| use deserr::DeserializeFromValue; | ||||
| use index_scheduler::{IndexScheduler, Query, TaskId}; | ||||
| use meilisearch_types::error::{ | ||||
|     deserr_codes::*, parse_option_cs_star_or, parse_option_u32_query_param, | ||||
|     parse_option_vec_u32_query_param, DeserrQueryParamError, DetailedParseIntError, | ||||
|     TakeErrorMessage, | ||||
| }; | ||||
| use meilisearch_types::error::{parse_u32_query_param, ResponseError}; | ||||
| use meilisearch_types::deserr::query_params::Param; | ||||
| use meilisearch_types::deserr::DeserrQueryParamError; | ||||
| use meilisearch_types::error::ResponseError; | ||||
| use meilisearch_types::error::{deserr_codes::*, InvalidTaskDateError}; | ||||
| use meilisearch_types::index_uid::IndexUid; | ||||
| use meilisearch_types::serde_cs; | ||||
| use meilisearch_types::settings::{Settings, Unchecked}; | ||||
| use meilisearch_types::star_or::StarOr; | ||||
| use meilisearch_types::star_or::{OptionStarOr, OptionStarOrList}; | ||||
| use meilisearch_types::tasks::{ | ||||
|     serialize_duration, Details, IndexSwap, Kind, KindWithContent, Status, Task, | ||||
| }; | ||||
| use serde::{Deserialize, Serialize}; | ||||
| use serde_cs::vec::CS; | ||||
| use serde::Serialize; | ||||
| use serde_json::json; | ||||
| use time::format_description::well_known::Rfc3339; | ||||
| use time::macros::format_description; | ||||
| @@ -30,7 +26,7 @@ use crate::extractors::authentication::GuardedData; | ||||
| use crate::extractors::query_parameters::QueryParameter; | ||||
| use crate::extractors::sequential_extractor::SeqHandler; | ||||
|  | ||||
| const DEFAULT_LIMIT: fn() -> u32 = || 20; | ||||
| const DEFAULT_LIMIT: u32 = 20; | ||||
|  | ||||
| pub fn configure(cfg: &mut web::ServiceConfig) { | ||||
|     cfg.service( | ||||
| @@ -169,62 +165,121 @@ impl From<Details> for DetailsView { | ||||
| #[derive(Debug, DeserializeFromValue)] | ||||
| #[deserr(error = DeserrQueryParamError, rename_all = camelCase, deny_unknown_fields)] | ||||
| pub struct TasksFilterQuery { | ||||
|     #[deserr(default = DEFAULT_LIMIT(), error = DeserrQueryParamError<InvalidTaskLimit>, from(String) = parse_u32_query_param -> TakeErrorMessage<DetailedParseIntError>)] | ||||
|     pub limit: u32, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskFrom>, from(Option<String>) = parse_option_u32_query_param -> TakeErrorMessage<DetailedParseIntError>)] | ||||
|     pub from: Option<TaskId>, | ||||
|     #[deserr(default = Param(DEFAULT_LIMIT), error = DeserrQueryParamError<InvalidTaskLimit>)] | ||||
|     pub limit: Param<u32>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskFrom>)] | ||||
|     pub from: Option<Param<TaskId>>, | ||||
|  | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskUids>, from(Option<CS<String>>) = parse_option_vec_u32_query_param -> TakeErrorMessage<DetailedParseIntError>)] | ||||
|     pub uids: Option<Vec<u32>>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskCanceledBy>, from(Option<CS<String>>) = parse_option_vec_u32_query_param -> TakeErrorMessage<DetailedParseIntError>)] | ||||
|     pub canceled_by: Option<Vec<u32>>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskTypes>, default = None, from(Option<CS<StarOr<String>>>) = parse_option_cs_star_or::<Kind> -> TakeErrorMessage<ResponseError>)] | ||||
|     pub types: Option<Vec<Kind>>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskStatuses>, default = None, from(Option<CS<StarOr<String>>>) = parse_option_cs_star_or::<Status> -> TakeErrorMessage<ResponseError>)] | ||||
|     pub statuses: Option<Vec<Status>>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidIndexUid>, default = None, from(Option<CS<StarOr<String>>>) = parse_option_cs_star_or::<IndexUid> -> TakeErrorMessage<ResponseError>)] | ||||
|     pub index_uids: Option<Vec<IndexUid>>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskUids>)] | ||||
|     pub uids: OptionStarOrList<u32>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskCanceledBy>)] | ||||
|     pub canceled_by: OptionStarOrList<u32>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskTypes>)] | ||||
|     pub types: OptionStarOrList<Kind>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskStatuses>)] | ||||
|     pub statuses: OptionStarOrList<Status>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidIndexUid>)] | ||||
|     pub index_uids: OptionStarOrList<IndexUid>, | ||||
|  | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskAfterEnqueuedAt>, default = None, from(Option<String>) = deserialize_date_after -> TakeErrorMessage<InvalidTaskDateError>)] | ||||
|     pub after_enqueued_at: Option<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskBeforeEnqueuedAt>, default = None, from(Option<String>) = deserialize_date_before -> TakeErrorMessage<InvalidTaskDateError>)] | ||||
|     pub before_enqueued_at: Option<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskAfterStartedAt>, default = None, from(Option<String>) = deserialize_date_after -> TakeErrorMessage<InvalidTaskDateError>)] | ||||
|     pub after_started_at: Option<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskBeforeStartedAt>, default = None, from(Option<String>) = deserialize_date_before -> TakeErrorMessage<InvalidTaskDateError>)] | ||||
|     pub before_started_at: Option<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskAfterFinishedAt>, default = None, from(Option<String>) = deserialize_date_after -> TakeErrorMessage<InvalidTaskDateError>)] | ||||
|     pub after_finished_at: Option<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskBeforeFinishedAt>, default = None, from(Option<String>) = deserialize_date_before -> TakeErrorMessage<InvalidTaskDateError>)] | ||||
|     pub before_finished_at: Option<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskAfterEnqueuedAt>, from(OptionStarOr<String>) = deserialize_date_after -> InvalidTaskDateError)] | ||||
|     pub after_enqueued_at: OptionStarOr<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskBeforeEnqueuedAt>, from(OptionStarOr<String>) = deserialize_date_before -> InvalidTaskDateError)] | ||||
|     pub before_enqueued_at: OptionStarOr<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskAfterStartedAt>, from(OptionStarOr<String>) = deserialize_date_after -> InvalidTaskDateError)] | ||||
|     pub after_started_at: OptionStarOr<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskBeforeStartedAt>, from(OptionStarOr<String>) = deserialize_date_before -> InvalidTaskDateError)] | ||||
|     pub before_started_at: OptionStarOr<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskAfterFinishedAt>, from(OptionStarOr<String>) = deserialize_date_after -> InvalidTaskDateError)] | ||||
|     pub after_finished_at: OptionStarOr<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskBeforeFinishedAt>, from(OptionStarOr<String>) = deserialize_date_before -> InvalidTaskDateError)] | ||||
|     pub before_finished_at: OptionStarOr<OffsetDateTime>, | ||||
| } | ||||
| impl TasksFilterQuery { | ||||
|     fn into_query(self) -> Query { | ||||
|         Query { | ||||
|             limit: Some(self.limit.0), | ||||
|             from: self.from.as_deref().copied(), | ||||
|             statuses: self.statuses.merge_star_and_none(), | ||||
|             types: self.types.merge_star_and_none(), | ||||
|             index_uids: self.index_uids.map(|x| x.to_string()).merge_star_and_none(), | ||||
|             uids: self.uids.merge_star_and_none(), | ||||
|             canceled_by: self.canceled_by.merge_star_and_none(), | ||||
|             before_enqueued_at: self.before_enqueued_at.merge_star_and_none(), | ||||
|             after_enqueued_at: self.after_enqueued_at.merge_star_and_none(), | ||||
|             before_started_at: self.before_started_at.merge_star_and_none(), | ||||
|             after_started_at: self.after_started_at.merge_star_and_none(), | ||||
|             before_finished_at: self.before_finished_at.merge_star_and_none(), | ||||
|             after_finished_at: self.after_finished_at.merge_star_and_none(), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Deserialize, Debug, DeserializeFromValue)] | ||||
| impl TaskDeletionOrCancelationQuery { | ||||
|     fn is_empty(&self) -> bool { | ||||
|         matches!( | ||||
|             self, | ||||
|             TaskDeletionOrCancelationQuery { | ||||
|                 uids: OptionStarOrList::None, | ||||
|                 canceled_by: OptionStarOrList::None, | ||||
|                 types: OptionStarOrList::None, | ||||
|                 statuses: OptionStarOrList::None, | ||||
|                 index_uids: OptionStarOrList::None, | ||||
|                 after_enqueued_at: OptionStarOr::None, | ||||
|                 before_enqueued_at: OptionStarOr::None, | ||||
|                 after_started_at: OptionStarOr::None, | ||||
|                 before_started_at: OptionStarOr::None, | ||||
|                 after_finished_at: OptionStarOr::None, | ||||
|                 before_finished_at: OptionStarOr::None | ||||
|             } | ||||
|         ) | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Debug, DeserializeFromValue)] | ||||
| #[deserr(error = DeserrQueryParamError, rename_all = camelCase, deny_unknown_fields)] | ||||
| pub struct TaskDeletionOrCancelationQuery { | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskUids>, from(Option<CS<String>>) = parse_option_vec_u32_query_param -> TakeErrorMessage<DetailedParseIntError>)] | ||||
|     pub uids: Option<Vec<u32>>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskCanceledBy>, from(Option<CS<String>>) = parse_option_vec_u32_query_param -> TakeErrorMessage<DetailedParseIntError>)] | ||||
|     pub canceled_by: Option<Vec<u32>>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskTypes>, default = None, from(Option<CS<StarOr<String>>>) = parse_option_cs_star_or::<Kind> -> TakeErrorMessage<ResponseError>)] | ||||
|     pub types: Option<Vec<Kind>>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskStatuses>, default = None, from(Option<CS<StarOr<String>>>) = parse_option_cs_star_or::<Status> -> TakeErrorMessage<ResponseError>)] | ||||
|     pub statuses: Option<Vec<Status>>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidIndexUid>, default = None, from(Option<CS<StarOr<String>>>) = parse_option_cs_star_or::<IndexUid> -> TakeErrorMessage<ResponseError>)] | ||||
|     pub index_uids: Option<Vec<IndexUid>>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskUids>)] | ||||
|     pub uids: OptionStarOrList<u32>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskCanceledBy>)] | ||||
|     pub canceled_by: OptionStarOrList<u32>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskTypes>)] | ||||
|     pub types: OptionStarOrList<Kind>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskStatuses>)] | ||||
|     pub statuses: OptionStarOrList<Status>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidIndexUid>)] | ||||
|     pub index_uids: OptionStarOrList<IndexUid>, | ||||
|  | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskAfterEnqueuedAt>, default = None, from(Option<String>) = deserialize_date_after -> TakeErrorMessage<InvalidTaskDateError>)] | ||||
|     pub after_enqueued_at: Option<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskBeforeEnqueuedAt>, default = None, from(Option<String>) = deserialize_date_before -> TakeErrorMessage<InvalidTaskDateError>)] | ||||
|     pub before_enqueued_at: Option<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskAfterStartedAt>, default = None, from(Option<String>) = deserialize_date_after -> TakeErrorMessage<InvalidTaskDateError>)] | ||||
|     pub after_started_at: Option<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskBeforeStartedAt>, default = None, from(Option<String>) = deserialize_date_before -> TakeErrorMessage<InvalidTaskDateError>)] | ||||
|     pub before_started_at: Option<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskAfterFinishedAt>, default = None, from(Option<String>) = deserialize_date_after -> TakeErrorMessage<InvalidTaskDateError>)] | ||||
|     pub after_finished_at: Option<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskBeforeFinishedAt>, default = None, from(Option<String>) = deserialize_date_before -> TakeErrorMessage<InvalidTaskDateError>)] | ||||
|     pub before_finished_at: Option<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskAfterEnqueuedAt>, from(OptionStarOr<String>) = deserialize_date_after -> InvalidTaskDateError)] | ||||
|     pub after_enqueued_at: OptionStarOr<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskBeforeEnqueuedAt>, from(OptionStarOr<String>) = deserialize_date_before -> InvalidTaskDateError)] | ||||
|     pub before_enqueued_at: OptionStarOr<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskAfterStartedAt>, from(OptionStarOr<String>) = deserialize_date_after -> InvalidTaskDateError)] | ||||
|     pub after_started_at: OptionStarOr<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskBeforeStartedAt>, from(OptionStarOr<String>) = deserialize_date_before -> InvalidTaskDateError)] | ||||
|     pub before_started_at: OptionStarOr<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskAfterFinishedAt>, from(OptionStarOr<String>) = deserialize_date_after -> InvalidTaskDateError)] | ||||
|     pub after_finished_at: OptionStarOr<OffsetDateTime>, | ||||
|     #[deserr(default, error = DeserrQueryParamError<InvalidTaskBeforeFinishedAt>, from(OptionStarOr<String>) = deserialize_date_before -> InvalidTaskDateError)] | ||||
|     pub before_finished_at: OptionStarOr<OffsetDateTime>, | ||||
| } | ||||
| impl TaskDeletionOrCancelationQuery { | ||||
|     fn into_query(self) -> Query { | ||||
|         Query { | ||||
|             limit: None, | ||||
|             from: None, | ||||
|             statuses: self.statuses.merge_star_and_none(), | ||||
|             types: self.types.merge_star_and_none(), | ||||
|             index_uids: self.index_uids.map(|x| x.to_string()).merge_star_and_none(), | ||||
|             uids: self.uids.merge_star_and_none(), | ||||
|             canceled_by: self.canceled_by.merge_star_and_none(), | ||||
|             before_enqueued_at: self.before_enqueued_at.merge_star_and_none(), | ||||
|             after_enqueued_at: self.after_enqueued_at.merge_star_and_none(), | ||||
|             before_started_at: self.before_started_at.merge_star_and_none(), | ||||
|             after_started_at: self.after_started_at.merge_star_and_none(), | ||||
|             before_finished_at: self.before_finished_at.merge_star_and_none(), | ||||
|             after_finished_at: self.after_finished_at.merge_star_and_none(), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| async fn cancel_tasks( | ||||
| @@ -233,57 +288,31 @@ async fn cancel_tasks( | ||||
|     req: HttpRequest, | ||||
|     analytics: web::Data<dyn Analytics>, | ||||
| ) -> Result<HttpResponse, ResponseError> { | ||||
|     let TaskDeletionOrCancelationQuery { | ||||
|         types, | ||||
|         uids, | ||||
|         canceled_by, | ||||
|         statuses, | ||||
|         index_uids, | ||||
|         after_enqueued_at, | ||||
|         before_enqueued_at, | ||||
|         after_started_at, | ||||
|         before_started_at, | ||||
|         after_finished_at, | ||||
|         before_finished_at, | ||||
|     } = params.into_inner(); | ||||
|     let params = params.into_inner(); | ||||
|  | ||||
|     if params.is_empty() { | ||||
|         return Err(index_scheduler::Error::TaskCancelationWithEmptyQuery.into()); | ||||
|     } | ||||
|  | ||||
|     analytics.publish( | ||||
|         "Tasks Canceled".to_string(), | ||||
|         json!({ | ||||
|             "filtered_by_uid": uids.is_some(), | ||||
|             "filtered_by_index_uid": index_uids.is_some(), | ||||
|             "filtered_by_type": types.is_some(), | ||||
|             "filtered_by_status": statuses.is_some(), | ||||
|             "filtered_by_canceled_by": canceled_by.is_some(), | ||||
|             "filtered_by_before_enqueued_at": before_enqueued_at.is_some(), | ||||
|             "filtered_by_after_enqueued_at": after_enqueued_at.is_some(), | ||||
|             "filtered_by_before_started_at": before_started_at.is_some(), | ||||
|             "filtered_by_after_started_at": after_started_at.is_some(), | ||||
|             "filtered_by_before_finished_at": before_finished_at.is_some(), | ||||
|             "filtered_by_after_finished_at": after_finished_at.is_some(), | ||||
|             "filtered_by_uid": params.uids.is_some(), | ||||
|             "filtered_by_index_uid": params.index_uids.is_some(), | ||||
|             "filtered_by_type": params.types.is_some(), | ||||
|             "filtered_by_status": params.statuses.is_some(), | ||||
|             "filtered_by_canceled_by": params.canceled_by.is_some(), | ||||
|             "filtered_by_before_enqueued_at": params.before_enqueued_at.is_some(), | ||||
|             "filtered_by_after_enqueued_at": params.after_enqueued_at.is_some(), | ||||
|             "filtered_by_before_started_at": params.before_started_at.is_some(), | ||||
|             "filtered_by_after_started_at": params.after_started_at.is_some(), | ||||
|             "filtered_by_before_finished_at": params.before_finished_at.is_some(), | ||||
|             "filtered_by_after_finished_at": params.after_finished_at.is_some(), | ||||
|         }), | ||||
|         Some(&req), | ||||
|     ); | ||||
|  | ||||
|     let query = Query { | ||||
|         limit: None, | ||||
|         from: None, | ||||
|         statuses, | ||||
|         types, | ||||
|         index_uids: index_uids.map(|xs| xs.into_iter().map(|s| s.to_string()).collect()), | ||||
|         uids, | ||||
|         canceled_by, | ||||
|         before_enqueued_at, | ||||
|         after_enqueued_at, | ||||
|         before_started_at, | ||||
|         after_started_at, | ||||
|         before_finished_at, | ||||
|         after_finished_at, | ||||
|     }; | ||||
|  | ||||
|     if query.is_empty() { | ||||
|         return Err(index_scheduler::Error::TaskCancelationWithEmptyQuery.into()); | ||||
|     } | ||||
|     let query = params.into_query(); | ||||
|  | ||||
|     let tasks = index_scheduler.get_task_ids_from_authorized_indexes( | ||||
|         &index_scheduler.read_txn()?, | ||||
| @@ -305,58 +334,30 @@ async fn delete_tasks( | ||||
|     req: HttpRequest, | ||||
|     analytics: web::Data<dyn Analytics>, | ||||
| ) -> Result<HttpResponse, ResponseError> { | ||||
|     let TaskDeletionOrCancelationQuery { | ||||
|         types, | ||||
|         uids, | ||||
|         canceled_by, | ||||
|         statuses, | ||||
|         index_uids, | ||||
|     let params = params.into_inner(); | ||||
|  | ||||
|         after_enqueued_at, | ||||
|         before_enqueued_at, | ||||
|         after_started_at, | ||||
|         before_started_at, | ||||
|         after_finished_at, | ||||
|         before_finished_at, | ||||
|     } = params.into_inner(); | ||||
|     if params.is_empty() { | ||||
|         return Err(index_scheduler::Error::TaskDeletionWithEmptyQuery.into()); | ||||
|     } | ||||
|  | ||||
|     analytics.publish( | ||||
|         "Tasks Deleted".to_string(), | ||||
|         json!({ | ||||
|             "filtered_by_uid": uids.is_some(), | ||||
|             "filtered_by_index_uid": index_uids.is_some(), | ||||
|             "filtered_by_type": types.is_some(), | ||||
|             "filtered_by_status": statuses.is_some(), | ||||
|             "filtered_by_canceled_by": canceled_by.is_some(), | ||||
|             "filtered_by_before_enqueued_at": before_enqueued_at.is_some(), | ||||
|             "filtered_by_after_enqueued_at": after_enqueued_at.is_some(), | ||||
|             "filtered_by_before_started_at": before_started_at.is_some(), | ||||
|             "filtered_by_after_started_at": after_started_at.is_some(), | ||||
|             "filtered_by_before_finished_at": before_finished_at.is_some(), | ||||
|             "filtered_by_after_finished_at": after_finished_at.is_some(), | ||||
|             "filtered_by_uid": params.uids.is_some(), | ||||
|             "filtered_by_index_uid": params.index_uids.is_some(), | ||||
|             "filtered_by_type": params.types.is_some(), | ||||
|             "filtered_by_status": params.statuses.is_some(), | ||||
|             "filtered_by_canceled_by": params.canceled_by.is_some(), | ||||
|             "filtered_by_before_enqueued_at": params.before_enqueued_at.is_some(), | ||||
|             "filtered_by_after_enqueued_at": params.after_enqueued_at.is_some(), | ||||
|             "filtered_by_before_started_at": params.before_started_at.is_some(), | ||||
|             "filtered_by_after_started_at": params.after_started_at.is_some(), | ||||
|             "filtered_by_before_finished_at": params.before_finished_at.is_some(), | ||||
|             "filtered_by_after_finished_at": params.after_finished_at.is_some(), | ||||
|         }), | ||||
|         Some(&req), | ||||
|     ); | ||||
|  | ||||
|     let query = Query { | ||||
|         limit: None, | ||||
|         from: None, | ||||
|         statuses, | ||||
|         types, | ||||
|         index_uids: index_uids.map(|xs| xs.into_iter().map(|s| s.to_string()).collect()), | ||||
|         uids, | ||||
|         canceled_by, | ||||
|         after_enqueued_at, | ||||
|         before_enqueued_at, | ||||
|         after_started_at, | ||||
|         before_started_at, | ||||
|         after_finished_at, | ||||
|         before_finished_at, | ||||
|     }; | ||||
|  | ||||
|     if query.is_empty() { | ||||
|         return Err(index_scheduler::Error::TaskDeletionWithEmptyQuery.into()); | ||||
|     } | ||||
|     let query = params.into_query(); | ||||
|  | ||||
|     let tasks = index_scheduler.get_task_ids_from_authorized_indexes( | ||||
|         &index_scheduler.read_txn()?, | ||||
| @@ -386,43 +387,13 @@ async fn get_tasks( | ||||
|     req: HttpRequest, | ||||
|     analytics: web::Data<dyn Analytics>, | ||||
| ) -> Result<HttpResponse, ResponseError> { | ||||
|     let params = params.into_inner(); | ||||
|     let mut params = params.into_inner(); | ||||
|     analytics.get_tasks(¶ms, &req); | ||||
|  | ||||
|     let TasksFilterQuery { | ||||
|         types, | ||||
|         uids, | ||||
|         canceled_by, | ||||
|         statuses, | ||||
|         index_uids, | ||||
|         limit, | ||||
|         from, | ||||
|         after_enqueued_at, | ||||
|         before_enqueued_at, | ||||
|         after_started_at, | ||||
|         before_started_at, | ||||
|         after_finished_at, | ||||
|         before_finished_at, | ||||
|     } = params; | ||||
|  | ||||
|     // We +1 just to know if there is more after this "page" or not. | ||||
|     let limit = limit.saturating_add(1); | ||||
|  | ||||
|     let query = index_scheduler::Query { | ||||
|         limit: Some(limit), | ||||
|         from, | ||||
|         statuses, | ||||
|         types, | ||||
|         index_uids: index_uids.map(|xs| xs.into_iter().map(|s| s.to_string()).collect()), | ||||
|         uids, | ||||
|         canceled_by, | ||||
|         before_enqueued_at, | ||||
|         after_enqueued_at, | ||||
|         before_started_at, | ||||
|         after_started_at, | ||||
|         before_finished_at, | ||||
|         after_finished_at, | ||||
|     }; | ||||
|     params.limit.0 = params.limit.0.saturating_add(1); | ||||
|     let limit = params.limit.0; | ||||
|     let query = params.into_query(); | ||||
|  | ||||
|     let mut tasks_results: Vec<TaskView> = index_scheduler | ||||
|         .get_tasks_from_authorized_indexes( | ||||
| @@ -488,7 +459,7 @@ pub enum DeserializeDateOption { | ||||
| pub fn deserialize_date( | ||||
|     value: &str, | ||||
|     option: DeserializeDateOption, | ||||
| ) -> std::result::Result<OffsetDateTime, TakeErrorMessage<InvalidTaskDateError>> { | ||||
| ) -> std::result::Result<OffsetDateTime, InvalidTaskDateError> { | ||||
|     // We can't parse using time's rfc3339 format, since then we won't know what part of the | ||||
|     // datetime was not explicitly specified, and thus we won't be able to increment it to the | ||||
|     // next step. | ||||
| @@ -510,45 +481,26 @@ pub fn deserialize_date( | ||||
|             } | ||||
|         } | ||||
|     } else { | ||||
|         Err(TakeErrorMessage(InvalidTaskDateError(value.to_owned()))) | ||||
|         Err(InvalidTaskDateError(value.to_owned())) | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn deserialize_date_before( | ||||
|     value: Option<String>, | ||||
| ) -> std::result::Result<Option<OffsetDateTime>, TakeErrorMessage<InvalidTaskDateError>> { | ||||
|     if let Some(value) = value { | ||||
|         let date = deserialize_date(&value, DeserializeDateOption::Before)?; | ||||
|         Ok(Some(date)) | ||||
|     } else { | ||||
|         Ok(None) | ||||
|     } | ||||
| } | ||||
| pub fn deserialize_date_after( | ||||
|     value: Option<String>, | ||||
| ) -> std::result::Result<Option<OffsetDateTime>, TakeErrorMessage<InvalidTaskDateError>> { | ||||
|     if let Some(value) = value { | ||||
|         let date = deserialize_date(&value, DeserializeDateOption::After)?; | ||||
|         Ok(Some(date)) | ||||
|     } else { | ||||
|         Ok(None) | ||||
|     value: OptionStarOr<String>, | ||||
| ) -> std::result::Result<OptionStarOr<OffsetDateTime>, InvalidTaskDateError> { | ||||
|     value.try_map(|x| deserialize_date(&x, DeserializeDateOption::After)) | ||||
| } | ||||
| pub fn deserialize_date_before( | ||||
|     value: OptionStarOr<String>, | ||||
| ) -> std::result::Result<OptionStarOr<OffsetDateTime>, InvalidTaskDateError> { | ||||
|     value.try_map(|x| deserialize_date(&x, DeserializeDateOption::Before)) | ||||
| } | ||||
|  | ||||
| #[derive(Debug)] | ||||
| pub struct InvalidTaskDateError(String); | ||||
| impl std::fmt::Display for InvalidTaskDateError { | ||||
|     fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||||
|         write!(f, "`{}` is an invalid date-time. It should follow the YYYY-MM-DD or RFC 3339 date-time format.", self.0) | ||||
|     } | ||||
| } | ||||
| impl std::error::Error for InvalidTaskDateError {} | ||||
|  | ||||
| #[cfg(test)] | ||||
| mod tests { | ||||
|     use deserr::DeserializeFromValue; | ||||
|     use meili_snap::snapshot; | ||||
|     use meilisearch_types::error::DeserrQueryParamError; | ||||
|     use meilisearch_types::deserr::DeserrQueryParamError; | ||||
|  | ||||
|     use crate::extractors::query_parameters::QueryParameter; | ||||
|     use crate::routes::tasks::{TaskDeletionOrCancelationQuery, TasksFilterQuery}; | ||||
| @@ -566,65 +518,71 @@ mod tests { | ||||
|             let params = "afterEnqueuedAt=2021-12-03&beforeEnqueuedAt=2021-12-03&afterStartedAt=2021-12-03&beforeStartedAt=2021-12-03&afterFinishedAt=2021-12-03&beforeFinishedAt=2021-12-03"; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|  | ||||
|             snapshot!(format!("{:?}", query.after_enqueued_at.unwrap()), @"2021-12-04 0:00:00.0 +00:00:00"); | ||||
|             snapshot!(format!("{:?}", query.before_enqueued_at.unwrap()), @"2021-12-03 0:00:00.0 +00:00:00"); | ||||
|             snapshot!(format!("{:?}", query.after_started_at.unwrap()), @"2021-12-04 0:00:00.0 +00:00:00"); | ||||
|             snapshot!(format!("{:?}", query.before_started_at.unwrap()), @"2021-12-03 0:00:00.0 +00:00:00"); | ||||
|             snapshot!(format!("{:?}", query.after_finished_at.unwrap()), @"2021-12-04 0:00:00.0 +00:00:00"); | ||||
|             snapshot!(format!("{:?}", query.before_finished_at.unwrap()), @"2021-12-03 0:00:00.0 +00:00:00"); | ||||
|             snapshot!(format!("{:?}", query.after_enqueued_at), @"Other(2021-12-04 0:00:00.0 +00:00:00)"); | ||||
|             snapshot!(format!("{:?}", query.before_enqueued_at), @"Other(2021-12-03 0:00:00.0 +00:00:00)"); | ||||
|             snapshot!(format!("{:?}", query.after_started_at), @"Other(2021-12-04 0:00:00.0 +00:00:00)"); | ||||
|             snapshot!(format!("{:?}", query.before_started_at), @"Other(2021-12-03 0:00:00.0 +00:00:00)"); | ||||
|             snapshot!(format!("{:?}", query.after_finished_at), @"Other(2021-12-04 0:00:00.0 +00:00:00)"); | ||||
|             snapshot!(format!("{:?}", query.before_finished_at), @"Other(2021-12-03 0:00:00.0 +00:00:00)"); | ||||
|         } | ||||
|         { | ||||
|             let params = | ||||
|                 "afterEnqueuedAt=2021-12-03T23:45:23Z&beforeEnqueuedAt=2021-12-03T23:45:23Z"; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|             snapshot!(format!("{:?}", query.after_enqueued_at.unwrap()), @"2021-12-03 23:45:23.0 +00:00:00"); | ||||
|             snapshot!(format!("{:?}", query.before_enqueued_at.unwrap()), @"2021-12-03 23:45:23.0 +00:00:00"); | ||||
|             snapshot!(format!("{:?}", query.after_enqueued_at), @"Other(2021-12-03 23:45:23.0 +00:00:00)"); | ||||
|             snapshot!(format!("{:?}", query.before_enqueued_at), @"Other(2021-12-03 23:45:23.0 +00:00:00)"); | ||||
|         } | ||||
|         { | ||||
|             let params = "afterEnqueuedAt=1997-11-12T09:55:06-06:20"; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|             snapshot!(format!("{:?}", query.after_enqueued_at.unwrap()), @"1997-11-12 9:55:06.0 -06:20:00"); | ||||
|             snapshot!(format!("{:?}", query.after_enqueued_at), @"Other(1997-11-12 9:55:06.0 -06:20:00)"); | ||||
|         } | ||||
|         { | ||||
|             let params = "afterEnqueuedAt=1997-11-12T09:55:06%2B00:00"; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|             snapshot!(format!("{:?}", query.after_enqueued_at.unwrap()), @"1997-11-12 9:55:06.0 +00:00:00"); | ||||
|             snapshot!(format!("{:?}", query.after_enqueued_at), @"Other(1997-11-12 9:55:06.0 +00:00:00)"); | ||||
|         } | ||||
|         { | ||||
|             let params = "afterEnqueuedAt=1997-11-12T09:55:06.200000300Z"; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|             snapshot!(format!("{:?}", query.after_enqueued_at.unwrap()), @"1997-11-12 9:55:06.2000003 +00:00:00"); | ||||
|             snapshot!(format!("{:?}", query.after_enqueued_at), @"Other(1997-11-12 9:55:06.2000003 +00:00:00)"); | ||||
|         } | ||||
|         { | ||||
|             // Stars are allowed in date fields as well | ||||
|             let params = "afterEnqueuedAt=*&beforeStartedAt=*&afterFinishedAt=*&beforeFinishedAt=*&afterStartedAt=*&beforeEnqueuedAt=*"; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|             snapshot!(format!("{:?}", query), @"TaskDeletionOrCancelationQuery { uids: None, canceled_by: None, types: None, statuses: None, index_uids: None, after_enqueued_at: Star, before_enqueued_at: Star, after_started_at: Star, before_started_at: Star, after_finished_at: Star, before_finished_at: Star }"); | ||||
|         } | ||||
|         { | ||||
|             let params = "afterFinishedAt=2021"; | ||||
|             let err = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap_err(); | ||||
|             snapshot!(format!("{err}"), @"Invalid value in parameter `afterFinishedAt`: `2021` is an invalid date-time. It should follow the YYYY-MM-DD or RFC 3339 date-time format."); | ||||
|             snapshot!(format!("{err:?}"), @r###"ResponseError { code: 400, message: "Invalid value in parameter `afterFinishedAt`: `2021` is an invalid date-time. It should follow the YYYY-MM-DD or RFC 3339 date-time format.", error_code: "invalid_task_after_finished_at", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#invalid-task-after-finished-at" }"###); | ||||
|         } | ||||
|         { | ||||
|             let params = "beforeFinishedAt=2021"; | ||||
|             let err = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap_err(); | ||||
|             snapshot!(format!("{err}"), @"Invalid value in parameter `beforeFinishedAt`: `2021` is an invalid date-time. It should follow the YYYY-MM-DD or RFC 3339 date-time format."); | ||||
|             snapshot!(format!("{err:?}"), @r###"ResponseError { code: 400, message: "Invalid value in parameter `beforeFinishedAt`: `2021` is an invalid date-time. It should follow the YYYY-MM-DD or RFC 3339 date-time format.", error_code: "invalid_task_before_finished_at", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#invalid-task-before-finished-at" }"###); | ||||
|         } | ||||
|         { | ||||
|             let params = "afterEnqueuedAt=2021-12"; | ||||
|             let err = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap_err(); | ||||
|             snapshot!(format!("{err}"), @"Invalid value in parameter `afterEnqueuedAt`: `2021-12` is an invalid date-time. It should follow the YYYY-MM-DD or RFC 3339 date-time format."); | ||||
|             snapshot!(format!("{err:?}"), @r###"ResponseError { code: 400, message: "Invalid value in parameter `afterEnqueuedAt`: `2021-12` is an invalid date-time. It should follow the YYYY-MM-DD or RFC 3339 date-time format.", error_code: "invalid_task_after_enqueued_at", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#invalid-task-after-enqueued-at" }"###); | ||||
|         } | ||||
|  | ||||
|         { | ||||
|             let params = "beforeEnqueuedAt=2021-12-03T23"; | ||||
|             let err = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap_err(); | ||||
|             snapshot!(format!("{err}"), @"Invalid value in parameter `beforeEnqueuedAt`: `2021-12-03T23` is an invalid date-time. It should follow the YYYY-MM-DD or RFC 3339 date-time format."); | ||||
|             snapshot!(format!("{err:?}"), @r###"ResponseError { code: 400, message: "Invalid value in parameter `beforeEnqueuedAt`: `2021-12-03T23` is an invalid date-time. It should follow the YYYY-MM-DD or RFC 3339 date-time format.", error_code: "invalid_task_before_enqueued_at", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#invalid-task-before-enqueued-at" }"###); | ||||
|         } | ||||
|         { | ||||
|             let params = "afterStartedAt=2021-12-03T23:45"; | ||||
|             let err = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap_err(); | ||||
|             snapshot!(format!("{err}"), @"Invalid value in parameter `afterStartedAt`: `2021-12-03T23:45` is an invalid date-time. It should follow the YYYY-MM-DD or RFC 3339 date-time format."); | ||||
|             snapshot!(format!("{err:?}"), @r###"ResponseError { code: 400, message: "Invalid value in parameter `afterStartedAt`: `2021-12-03T23:45` is an invalid date-time. It should follow the YYYY-MM-DD or RFC 3339 date-time format.", error_code: "invalid_task_after_started_at", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#invalid-task-after-started-at" }"###); | ||||
|         } | ||||
|         { | ||||
|             let params = "beforeStartedAt=2021-12-03T23:45"; | ||||
|             let err = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap_err(); | ||||
|             snapshot!(format!("{err}"), @"Invalid value in parameter `beforeStartedAt`: `2021-12-03T23:45` is an invalid date-time. It should follow the YYYY-MM-DD or RFC 3339 date-time format."); | ||||
|             snapshot!(format!("{err:?}"), @r###"ResponseError { code: 400, message: "Invalid value in parameter `beforeStartedAt`: `2021-12-03T23:45` is an invalid date-time. It should follow the YYYY-MM-DD or RFC 3339 date-time format.", error_code: "invalid_task_before_started_at", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#invalid-task-before-started-at" }"###); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -633,22 +591,27 @@ mod tests { | ||||
|         { | ||||
|             let params = "uids=78,1,12,73"; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|             snapshot!(format!("{:?}", query.uids.unwrap()), @"[78, 1, 12, 73]"); | ||||
|             snapshot!(format!("{:?}", query.uids), @"List([78, 1, 12, 73])"); | ||||
|         } | ||||
|         { | ||||
|             let params = "uids=1"; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|             snapshot!(format!("{:?}", query.uids.unwrap()), @"[1]"); | ||||
|             snapshot!(format!("{:?}", query.uids), @"List([1])"); | ||||
|         } | ||||
|         { | ||||
|             let params = "uids=cat,*,dog"; | ||||
|             let err = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap_err(); | ||||
|             snapshot!(format!("{err:?}"), @r###"ResponseError { code: 400, message: "Invalid value in parameter `uids[0]`: could not parse `cat` as a positive integer", error_code: "invalid_task_uids", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#invalid-task-uids" }"###); | ||||
|         } | ||||
|         { | ||||
|             let params = "uids=78,hello,world"; | ||||
|             let err = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap_err(); | ||||
|             snapshot!(format!("{err}"), @"Invalid value in parameter `uids`: could not parse `hello` as a positive integer"); | ||||
|             snapshot!(format!("{err:?}"), @r###"ResponseError { code: 400, message: "Invalid value in parameter `uids[1]`: could not parse `hello` as a positive integer", error_code: "invalid_task_uids", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#invalid-task-uids" }"###); | ||||
|         } | ||||
|         { | ||||
|             let params = "uids=cat"; | ||||
|             let err = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap_err(); | ||||
|             snapshot!(format!("{err}"), @"Invalid value in parameter `uids`: could not parse `cat` as a positive integer"); | ||||
|             snapshot!(format!("{err:?}"), @r###"ResponseError { code: 400, message: "Invalid value in parameter `uids`: could not parse `cat` as a positive integer", error_code: "invalid_task_uids", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#invalid-task-uids" }"###); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -657,17 +620,17 @@ mod tests { | ||||
|         { | ||||
|             let params = "statuses=succeeded,failed,enqueued,processing,canceled"; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|             snapshot!(format!("{:?}", query.statuses.unwrap()), @"[Succeeded, Failed, Enqueued, Processing, Canceled]"); | ||||
|             snapshot!(format!("{:?}", query.statuses), @"List([Succeeded, Failed, Enqueued, Processing, Canceled])"); | ||||
|         } | ||||
|         { | ||||
|             let params = "statuses=enqueued"; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|             snapshot!(format!("{:?}", query.statuses.unwrap()), @"[Enqueued]"); | ||||
|             snapshot!(format!("{:?}", query.statuses), @"List([Enqueued])"); | ||||
|         } | ||||
|         { | ||||
|             let params = "statuses=finished"; | ||||
|             let err = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap_err(); | ||||
|             snapshot!(format!("{err}"), @"Invalid value in parameter `statuses`: `finished` is not a valid task status. Available statuses are `enqueued`, `processing`, `succeeded`, `failed`, `canceled`."); | ||||
|             snapshot!(format!("{err:?}"), @r###"ResponseError { code: 400, message: "Invalid value in parameter `statuses`: `finished` is not a valid task status. Available statuses are `enqueued`, `processing`, `succeeded`, `failed`, `canceled`.", error_code: "invalid_task_statuses", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#invalid-task-statuses" }"###); | ||||
|         } | ||||
|     } | ||||
|     #[test] | ||||
| @@ -675,17 +638,17 @@ mod tests { | ||||
|         { | ||||
|             let params = "types=documentAdditionOrUpdate,documentDeletion,settingsUpdate,indexCreation,indexDeletion,indexUpdate,indexSwap,taskCancelation,taskDeletion,dumpCreation,snapshotCreation"; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|             snapshot!(format!("{:?}", query.types.unwrap()), @"[DocumentAdditionOrUpdate, DocumentDeletion, SettingsUpdate, IndexCreation, IndexDeletion, IndexUpdate, IndexSwap, TaskCancelation, TaskDeletion, DumpCreation, SnapshotCreation]"); | ||||
|             snapshot!(format!("{:?}", query.types), @"List([DocumentAdditionOrUpdate, DocumentDeletion, SettingsUpdate, IndexCreation, IndexDeletion, IndexUpdate, IndexSwap, TaskCancelation, TaskDeletion, DumpCreation, SnapshotCreation])"); | ||||
|         } | ||||
|         { | ||||
|             let params = "types=settingsUpdate"; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|             snapshot!(format!("{:?}", query.types.unwrap()), @"[SettingsUpdate]"); | ||||
|             snapshot!(format!("{:?}", query.types), @"List([SettingsUpdate])"); | ||||
|         } | ||||
|         { | ||||
|             let params = "types=createIndex"; | ||||
|             let err = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap_err(); | ||||
|             snapshot!(format!("{err}"), @"Invalid value in parameter `types`: `createIndex` is not a valid task type. Available types are `documentAdditionOrUpdate`, `documentDeletion`, `settingsUpdate`, `indexCreation`, `indexDeletion`, `indexUpdate`, `indexSwap`, `taskCancelation`, `taskDeletion`, `dumpCreation`, `snapshotCreation`."); | ||||
|             snapshot!(format!("{err:?}"), @r###"ResponseError { code: 400, message: "Invalid value in parameter `types`: `createIndex` is not a valid task type. Available types are `documentAdditionOrUpdate`, `documentDeletion`, `settingsUpdate`, `indexCreation`, `indexDeletion`, `indexUpdate`, `indexSwap`, `taskCancelation`, `taskDeletion`, `dumpCreation`, `snapshotCreation`.", error_code: "invalid_task_types", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#invalid-task-types" }"###); | ||||
|         } | ||||
|     } | ||||
|     #[test] | ||||
| @@ -693,22 +656,22 @@ mod tests { | ||||
|         { | ||||
|             let params = "indexUids=toto,tata-78"; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|             snapshot!(format!("{:?}", query.index_uids.unwrap()), @r###"[IndexUid("toto"), IndexUid("tata-78")]"###); | ||||
|             snapshot!(format!("{:?}", query.index_uids), @r###"List([IndexUid("toto"), IndexUid("tata-78")])"###); | ||||
|         } | ||||
|         { | ||||
|             let params = "indexUids=index_a"; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|             snapshot!(format!("{:?}", query.index_uids.unwrap()), @r###"[IndexUid("index_a")]"###); | ||||
|             snapshot!(format!("{:?}", query.index_uids), @r###"List([IndexUid("index_a")])"###); | ||||
|         } | ||||
|         { | ||||
|             let params = "indexUids=1,hé"; | ||||
|             let err = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap_err(); | ||||
|             snapshot!(format!("{err}"), @"Invalid value in parameter `indexUids`: `hé` is not a valid index uid. Index uid can be an integer or a string containing only alphanumeric characters, hyphens (-) and underscores (_)."); | ||||
|             snapshot!(format!("{err:?}"), @r###"ResponseError { code: 400, message: "Invalid value in parameter `indexUids[1]`: `hé` is not a valid index uid. Index uid can be an integer or a string containing only alphanumeric characters, hyphens (-) and underscores (_).", error_code: "invalid_index_uid", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#invalid-index-uid" }"###); | ||||
|         } | ||||
|         { | ||||
|             let params = "indexUids=hé"; | ||||
|             let err = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap_err(); | ||||
|             snapshot!(format!("{err}"), @"Invalid value in parameter `indexUids`: `hé` is not a valid index uid. Index uid can be an integer or a string containing only alphanumeric characters, hyphens (-) and underscores (_)."); | ||||
|             snapshot!(format!("{err:?}"), @r###"ResponseError { code: 400, message: "Invalid value in parameter `indexUids`: `hé` is not a valid index uid. Index uid can be an integer or a string containing only alphanumeric characters, hyphens (-) and underscores (_).", error_code: "invalid_index_uid", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#invalid-index-uid" }"###); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -717,38 +680,53 @@ mod tests { | ||||
|         { | ||||
|             let params = "from=12&limit=15&indexUids=toto,tata-78&statuses=succeeded,enqueued&afterEnqueuedAt=2012-04-23&uids=1,2,3"; | ||||
|             let query = deserr_query_params::<TasksFilterQuery>(params).unwrap(); | ||||
|             snapshot!(format!("{:?}", query), @r###"TasksFilterQuery { limit: 15, from: Some(12), uids: Some([1, 2, 3]), canceled_by: None, types: None, statuses: Some([Succeeded, Enqueued]), index_uids: Some([IndexUid("toto"), IndexUid("tata-78")]), after_enqueued_at: Some(2012-04-24 0:00:00.0 +00:00:00), before_enqueued_at: None, after_started_at: None, before_started_at: None, after_finished_at: None, before_finished_at: None }"###); | ||||
|             snapshot!(format!("{:?}", query), @r###"TasksFilterQuery { limit: Param(15), from: Some(Param(12)), uids: List([1, 2, 3]), canceled_by: None, types: None, statuses: List([Succeeded, Enqueued]), index_uids: List([IndexUid("toto"), IndexUid("tata-78")]), after_enqueued_at: Other(2012-04-24 0:00:00.0 +00:00:00), before_enqueued_at: None, after_started_at: None, before_started_at: None, after_finished_at: None, before_finished_at: None }"###); | ||||
|         } | ||||
|         { | ||||
|             // Stars should translate to `None` in the query | ||||
|             // Verify value of the default limit | ||||
|             let params = "indexUids=*&statuses=succeeded,*&afterEnqueuedAt=2012-04-23&uids=1,2,3"; | ||||
|             let query = deserr_query_params::<TasksFilterQuery>(params).unwrap(); | ||||
|             snapshot!(format!("{:?}", query), @"TasksFilterQuery { limit: 20, from: None, uids: Some([1, 2, 3]), canceled_by: None, types: None, statuses: None, index_uids: None, after_enqueued_at: Some(2012-04-24 0:00:00.0 +00:00:00), before_enqueued_at: None, after_started_at: None, before_started_at: None, after_finished_at: None, before_finished_at: None }"); | ||||
|             snapshot!(format!("{:?}", query), @"TasksFilterQuery { limit: Param(20), from: None, uids: List([1, 2, 3]), canceled_by: None, types: None, statuses: Star, index_uids: Star, after_enqueued_at: Other(2012-04-24 0:00:00.0 +00:00:00), before_enqueued_at: None, after_started_at: None, before_started_at: None, after_finished_at: None, before_finished_at: None }"); | ||||
|         } | ||||
|         { | ||||
|             // Stars should also translate to `None` in task deletion/cancelation queries | ||||
|             let params = "indexUids=*&statuses=succeeded,*&afterEnqueuedAt=2012-04-23&uids=1,2,3"; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|             snapshot!(format!("{:?}", query), @"TaskDeletionOrCancelationQuery { uids: Some([1, 2, 3]), canceled_by: None, types: None, statuses: None, index_uids: None, after_enqueued_at: Some(2012-04-24 0:00:00.0 +00:00:00), before_enqueued_at: None, after_started_at: None, before_started_at: None, after_finished_at: None, before_finished_at: None }"); | ||||
|             snapshot!(format!("{:?}", query), @"TaskDeletionOrCancelationQuery { uids: List([1, 2, 3]), canceled_by: None, types: None, statuses: Star, index_uids: Star, after_enqueued_at: Other(2012-04-24 0:00:00.0 +00:00:00), before_enqueued_at: None, after_started_at: None, before_started_at: None, after_finished_at: None, before_finished_at: None }"); | ||||
|         } | ||||
|         { | ||||
|             // Stars in uids not allowed | ||||
|             let params = "uids=*"; | ||||
|             // Star in from not allowed | ||||
|             let params = "uids=*&from=*"; | ||||
|             let err = deserr_query_params::<TasksFilterQuery>(params).unwrap_err(); | ||||
|             snapshot!(format!("{err}"), @"Invalid value in parameter `uids`: could not parse `*` as a positive integer"); | ||||
|             snapshot!(format!("{err:?}"), @r###"ResponseError { code: 400, message: "Invalid value in parameter `from`: could not parse `*` as a positive integer", error_code: "invalid_task_from", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#invalid-task-from" }"###); | ||||
|         } | ||||
|         { | ||||
|             // From not allowed in task deletion/cancelation queries | ||||
|             let params = "from=12"; | ||||
|             let err = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap_err(); | ||||
|             snapshot!(format!("{err}"), @"Unknown parameter `from`: expected one of `uids`, `canceledBy`, `types`, `statuses`, `indexUids`, `afterEnqueuedAt`, `beforeEnqueuedAt`, `afterStartedAt`, `beforeStartedAt`, `afterFinishedAt`, `beforeFinishedAt`"); | ||||
|             snapshot!(format!("{err:?}"), @r###"ResponseError { code: 400, message: "Unknown parameter `from`: expected one of `uids`, `canceledBy`, `types`, `statuses`, `indexUids`, `afterEnqueuedAt`, `beforeEnqueuedAt`, `afterStartedAt`, `beforeStartedAt`, `afterFinishedAt`, `beforeFinishedAt`", error_code: "bad_request", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#bad-request" }"###); | ||||
|         } | ||||
|         { | ||||
|             // Limit not allowed in task deletion/cancelation queries | ||||
|             let params = "limit=12"; | ||||
|             let err = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap_err(); | ||||
|             snapshot!(format!("{err}"), @"Unknown parameter `limit`: expected one of `uids`, `canceledBy`, `types`, `statuses`, `indexUids`, `afterEnqueuedAt`, `beforeEnqueuedAt`, `afterStartedAt`, `beforeStartedAt`, `afterFinishedAt`, `beforeFinishedAt`"); | ||||
|             snapshot!(format!("{err:?}"), @r###"ResponseError { code: 400, message: "Unknown parameter `limit`: expected one of `uids`, `canceledBy`, `types`, `statuses`, `indexUids`, `afterEnqueuedAt`, `beforeEnqueuedAt`, `afterStartedAt`, `beforeStartedAt`, `afterFinishedAt`, `beforeFinishedAt`", error_code: "bad_request", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#bad-request" }"###); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn deserialize_task_delete_or_cancel_empty() { | ||||
|         { | ||||
|             let params = ""; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|             assert!(query.is_empty()); | ||||
|         } | ||||
|         { | ||||
|             let params = "statuses=*"; | ||||
|             let query = deserr_query_params::<TaskDeletionOrCancelationQuery>(params).unwrap(); | ||||
|             assert!(!query.is_empty()); | ||||
|             snapshot!(format!("{query:?}"), @"TaskDeletionOrCancelationQuery { uids: None, canceled_by: None, types: None, statuses: Star, index_uids: None, after_enqueued_at: None, before_enqueued_at: None, after_started_at: None, before_started_at: None, after_finished_at: None, before_finished_at: None }"); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -5,8 +5,8 @@ use std::time::Instant; | ||||
|  | ||||
| use deserr::DeserializeFromValue; | ||||
| use either::Either; | ||||
| use meilisearch_types::deserr::DeserrJsonError; | ||||
| use meilisearch_types::error::deserr_codes::*; | ||||
| use meilisearch_types::error::DeserrJsonError; | ||||
| use meilisearch_types::settings::DEFAULT_PAGINATION_MAX_TOTAL_HITS; | ||||
| use meilisearch_types::{milli, Document}; | ||||
| use milli::tokenizer::TokenizerBuilder; | ||||
| @@ -15,7 +15,7 @@ use milli::{ | ||||
|     SortError, TermsMatchingStrategy, DEFAULT_VALUES_PER_FACET, | ||||
| }; | ||||
| use regex::Regex; | ||||
| use serde::{Deserialize, Serialize}; | ||||
| use serde::Serialize; | ||||
| use serde_json::{json, Value}; | ||||
|  | ||||
| use crate::error::MeilisearchHttpError; | ||||
| @@ -74,9 +74,8 @@ impl SearchQuery { | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Deserialize, Debug, Clone, PartialEq, Eq, DeserializeFromValue)] | ||||
| #[derive(Debug, Clone, PartialEq, Eq, DeserializeFromValue)] | ||||
| #[deserr(rename_all = camelCase)] | ||||
| #[serde(rename_all = "camelCase")] | ||||
| pub enum MatchingStrategy { | ||||
|     /// Remove query words from last to first | ||||
|     Last, | ||||
|   | ||||
| @@ -295,7 +295,7 @@ async fn search_bad_show_matches_position() { | ||||
|     snapshot!(code, @"400 Bad Request"); | ||||
|     snapshot!(json_string!(response), @r###" | ||||
|     { | ||||
|       "message": "Invalid value in parameter `showMatchesPosition`: provided string was not `true` or `false`", | ||||
|       "message": "Invalid value in parameter `showMatchesPosition`: could not parse `doggo` as a boolean, expected either `true` or `false`", | ||||
|       "code": "invalid_search_show_matches_position", | ||||
|       "type": "invalid_request", | ||||
|       "link": "https://docs.meilisearch.com/errors#invalid-search-show-matches-position" | ||||
|   | ||||
| @@ -43,7 +43,7 @@ async fn task_bad_uids() { | ||||
|     snapshot!(code, @"400 Bad Request"); | ||||
|     snapshot!(json_string!(response), @r###" | ||||
|     { | ||||
|       "message": "Invalid value in parameter `uids`: could not parse `dogo` as a positive integer", | ||||
|       "message": "Invalid value in parameter `uids[1]`: could not parse `dogo` as a positive integer", | ||||
|       "code": "invalid_task_uids", | ||||
|       "type": "invalid_request", | ||||
|       "link": "https://docs.meilisearch.com/errors#invalid-task-uids" | ||||
|   | ||||
		Reference in New Issue
	
	Block a user