mirror of
				https://github.com/meilisearch/meilisearch.git
				synced 2025-10-26 21:46:27 +00:00 
			
		
		
		
	Merge branch 'main' into bring-back-changes-v1.1.0
This commit is contained in:
		| @@ -81,6 +81,8 @@ impl FromStr for Member { | ||||
|                 if is_reserved_keyword(text) | ||||
|                     || text.starts_with("_geoRadius(") | ||||
|                     || text.starts_with("_geoBoundingBox(") | ||||
|                     || text.starts_with("_geo(") | ||||
|                     || text.starts_with("_geoDistance(") | ||||
|                 { | ||||
|                     return Err(AscDescError::ReservedKeyword { name: text.to_string() })?; | ||||
|                 } | ||||
| @@ -265,6 +267,13 @@ mod tests { | ||||
|             ("_geoPoint(0, -180.000001):desc", GeoError(BadGeoError::Lng(-180.000001))), | ||||
|             ("_geoPoint(159.256, 130):asc", GeoError(BadGeoError::Lat(159.256))), | ||||
|             ("_geoPoint(12, -2021):desc", GeoError(BadGeoError::Lng(-2021.))), | ||||
|             ("_geo(12, -2021):asc", ReservedKeyword { name: S("_geo(12, -2021)") }), | ||||
|             ("_geo(12, -2021):desc", ReservedKeyword { name: S("_geo(12, -2021)") }), | ||||
|             ("_geoDistance(12, -2021):asc", ReservedKeyword { name: S("_geoDistance(12, -2021)") }), | ||||
|             ( | ||||
|                 "_geoDistance(12, -2021):desc", | ||||
|                 ReservedKeyword { name: S("_geoDistance(12, -2021)") }, | ||||
|             ), | ||||
|         ]; | ||||
|  | ||||
|         for (req, expected_error) in invalid_req { | ||||
|   | ||||
| @@ -114,14 +114,15 @@ impl<W: Write> DocumentsBatchBuilder<W> { | ||||
|                 self.value_buffer.clear(); | ||||
|  | ||||
|                 let value = &record[*i]; | ||||
|                 let trimmed_value = value.trim(); | ||||
|                 match type_ { | ||||
|                     AllowedType::Number => { | ||||
|                         if value.trim().is_empty() { | ||||
|                         if trimmed_value.is_empty() { | ||||
|                             to_writer(&mut self.value_buffer, &Value::Null)?; | ||||
|                         } else if let Ok(integer) = value.trim().parse::<i64>() { | ||||
|                         } else if let Ok(integer) = trimmed_value.parse::<i64>() { | ||||
|                             to_writer(&mut self.value_buffer, &integer)?; | ||||
|                         } else { | ||||
|                             match value.trim().parse::<f64>() { | ||||
|                             match trimmed_value.parse::<f64>() { | ||||
|                                 Ok(float) => { | ||||
|                                     to_writer(&mut self.value_buffer, &float)?; | ||||
|                                 } | ||||
| @@ -135,6 +136,24 @@ impl<W: Write> DocumentsBatchBuilder<W> { | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                     AllowedType::Boolean => { | ||||
|                         if trimmed_value.is_empty() { | ||||
|                             to_writer(&mut self.value_buffer, &Value::Null)?; | ||||
|                         } else { | ||||
|                             match trimmed_value.parse::<bool>() { | ||||
|                                 Ok(bool) => { | ||||
|                                     to_writer(&mut self.value_buffer, &bool)?; | ||||
|                                 } | ||||
|                                 Err(error) => { | ||||
|                                     return Err(Error::ParseBool { | ||||
|                                         error, | ||||
|                                         line, | ||||
|                                         value: value.to_string(), | ||||
|                                     }); | ||||
|                                 } | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                     AllowedType::String => { | ||||
|                         if value.is_empty() { | ||||
|                             to_writer(&mut self.value_buffer, &Value::Null)?; | ||||
| @@ -173,6 +192,7 @@ impl<W: Write> DocumentsBatchBuilder<W> { | ||||
| #[derive(Debug)] | ||||
| enum AllowedType { | ||||
|     String, | ||||
|     Boolean, | ||||
|     Number, | ||||
| } | ||||
|  | ||||
| @@ -181,6 +201,7 @@ fn parse_csv_header(header: &str) -> (&str, AllowedType) { | ||||
|     match header.rsplit_once(':') { | ||||
|         Some((field_name, field_type)) => match field_type { | ||||
|             "string" => (field_name, AllowedType::String), | ||||
|             "boolean" => (field_name, AllowedType::Boolean), | ||||
|             "number" => (field_name, AllowedType::Number), | ||||
|             // if the pattern isn't reconized, we keep the whole field. | ||||
|             _otherwise => (header, AllowedType::String), | ||||
|   | ||||
| @@ -3,7 +3,7 @@ mod enriched; | ||||
| mod reader; | ||||
| mod serde_impl; | ||||
|  | ||||
| use std::fmt::{self, Debug}; | ||||
| use std::fmt::Debug; | ||||
| use std::io; | ||||
| use std::str::Utf8Error; | ||||
|  | ||||
| @@ -87,71 +87,30 @@ impl DocumentsBatchIndex { | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Debug)] | ||||
| #[derive(Debug, thiserror::Error)] | ||||
| pub enum Error { | ||||
|     #[error("Error parsing number {value:?} at line {line}: {error}")] | ||||
|     ParseFloat { error: std::num::ParseFloatError, line: usize, value: String }, | ||||
|     #[error("Error parsing boolean {value:?} at line {line}: {error}")] | ||||
|     ParseBool { error: std::str::ParseBoolError, line: usize, value: String }, | ||||
|     #[error("Invalid document addition format, missing the documents batch index.")] | ||||
|     InvalidDocumentFormat, | ||||
|     #[error("Invalid enriched data.")] | ||||
|     InvalidEnrichedData, | ||||
|     InvalidUtf8(Utf8Error), | ||||
|     Csv(csv::Error), | ||||
|     Json(serde_json::Error), | ||||
|     #[error(transparent)] | ||||
|     InvalidUtf8(#[from] Utf8Error), | ||||
|     #[error(transparent)] | ||||
|     Csv(#[from] csv::Error), | ||||
|     #[error(transparent)] | ||||
|     Json(#[from] serde_json::Error), | ||||
|     #[error(transparent)] | ||||
|     Serialize(serde_json::Error), | ||||
|     Grenad(grenad::Error), | ||||
|     Io(io::Error), | ||||
|     #[error(transparent)] | ||||
|     Grenad(#[from] grenad::Error), | ||||
|     #[error(transparent)] | ||||
|     Io(#[from] io::Error), | ||||
| } | ||||
|  | ||||
| impl From<csv::Error> for Error { | ||||
|     fn from(e: csv::Error) -> Self { | ||||
|         Self::Csv(e) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl From<io::Error> for Error { | ||||
|     fn from(other: io::Error) -> Self { | ||||
|         Self::Io(other) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl From<serde_json::Error> for Error { | ||||
|     fn from(other: serde_json::Error) -> Self { | ||||
|         Self::Json(other) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl From<grenad::Error> for Error { | ||||
|     fn from(other: grenad::Error) -> Self { | ||||
|         Self::Grenad(other) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl From<Utf8Error> for Error { | ||||
|     fn from(other: Utf8Error) -> Self { | ||||
|         Self::InvalidUtf8(other) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl fmt::Display for Error { | ||||
|     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||||
|         match self { | ||||
|             Error::ParseFloat { error, line, value } => { | ||||
|                 write!(f, "Error parsing number {:?} at line {}: {}", value, line, error) | ||||
|             } | ||||
|             Error::InvalidDocumentFormat => { | ||||
|                 f.write_str("Invalid document addition format, missing the documents batch index.") | ||||
|             } | ||||
|             Error::InvalidEnrichedData => f.write_str("Invalid enriched data."), | ||||
|             Error::InvalidUtf8(e) => write!(f, "{}", e), | ||||
|             Error::Io(e) => write!(f, "{}", e), | ||||
|             Error::Serialize(e) => write!(f, "{}", e), | ||||
|             Error::Grenad(e) => write!(f, "{}", e), | ||||
|             Error::Csv(e) => write!(f, "{}", e), | ||||
|             Error::Json(e) => write!(f, "{}", e), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl std::error::Error for Error {} | ||||
|  | ||||
| #[cfg(test)] | ||||
| pub fn objects_from_json_value(json: serde_json::Value) -> Vec<crate::Object> { | ||||
|     let documents = match json { | ||||
| @@ -274,6 +233,19 @@ mod test { | ||||
|         ]); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn csv_types_dont_panic() { | ||||
|         let csv1_content = | ||||
|             "id:number,b:boolean,c,d:number\n1,,,\n2,true,doggo,2\n3,false,the best doggo,-2\n4,,\"Hello, World!\",2.5"; | ||||
|         let csv1 = csv::Reader::from_reader(Cursor::new(csv1_content)); | ||||
|  | ||||
|         let mut builder = DocumentsBatchBuilder::new(Vec::new()); | ||||
|         builder.append_csv(csv1).unwrap(); | ||||
|         let vector = builder.into_inner().unwrap(); | ||||
|  | ||||
|         DocumentsBatchReader::from_reader(Cursor::new(vector)).unwrap(); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn out_of_order_csv_fields() { | ||||
|         let csv1_content = "id:number,b\n1,0"; | ||||
|   | ||||
| @@ -565,8 +565,12 @@ impl<'a, 't, 'u, 'i> Settings<'a, 't, 'u, 'i> { | ||||
|                     self.index.put_primary_key(self.wtxn, primary_key)?; | ||||
|                     Ok(()) | ||||
|                 } else { | ||||
|                     let primary_key = self.index.primary_key(self.wtxn)?.unwrap(); | ||||
|                     Err(UserError::PrimaryKeyCannotBeChanged(primary_key.to_string()).into()) | ||||
|                     let curr_primary_key = self.index.primary_key(self.wtxn)?.unwrap().to_string(); | ||||
|                     if primary_key == &curr_primary_key { | ||||
|                         Ok(()) | ||||
|                     } else { | ||||
|                         Err(UserError::PrimaryKeyCannotBeChanged(curr_primary_key).into()) | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             Setting::Reset => { | ||||
| @@ -1332,6 +1336,17 @@ mod tests { | ||||
|             .unwrap(); | ||||
|         wtxn.commit().unwrap(); | ||||
|  | ||||
|         // Updating settings with the same primary key should do nothing | ||||
|         let mut wtxn = index.write_txn().unwrap(); | ||||
|         index | ||||
|             .update_settings_using_wtxn(&mut wtxn, |settings| { | ||||
|                 settings.set_primary_key(S("mykey")); | ||||
|             }) | ||||
|             .unwrap(); | ||||
|         assert_eq!(index.primary_key(&wtxn).unwrap(), Some("mykey")); | ||||
|         wtxn.commit().unwrap(); | ||||
|  | ||||
|         // Updating the settings with a different (or no) primary key causes an error | ||||
|         let mut wtxn = index.write_txn().unwrap(); | ||||
|         let error = index | ||||
|             .update_settings_using_wtxn(&mut wtxn, |settings| { | ||||
|   | ||||
		Reference in New Issue
	
	Block a user