mirror of
				https://github.com/meilisearch/meilisearch.git
				synced 2025-10-25 21:16:28 +00:00 
			
		
		
		
	refactor schema
This commit is contained in:
		| @@ -6,22 +6,14 @@ use serde::{Deserialize, Serialize}; | |||||||
| use crate::{SResult, FieldId}; | use crate::{SResult, FieldId}; | ||||||
|  |  | ||||||
| #[derive(Debug, Default, Clone, PartialEq, Eq, Serialize, Deserialize)] | #[derive(Debug, Default, Clone, PartialEq, Eq, Serialize, Deserialize)] | ||||||
| pub struct FieldsMap { | pub(crate) struct FieldsMap { | ||||||
|     name_map: HashMap<String, FieldId>, |     name_map: HashMap<String, FieldId>, | ||||||
|     id_map: HashMap<FieldId, String>, |     id_map: HashMap<FieldId, String>, | ||||||
|     next_id: FieldId |     next_id: FieldId | ||||||
| } | } | ||||||
|  |  | ||||||
| impl FieldsMap { | impl FieldsMap { | ||||||
|     pub fn len(&self) -> usize { |     pub(crate) fn insert(&mut self, name: &str) -> SResult<FieldId> { | ||||||
|         self.name_map.len() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn is_empty(&self) -> bool { |  | ||||||
|         self.name_map.is_empty() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn insert(&mut self, name: &str) -> SResult<FieldId> { |  | ||||||
|         if let Some(id) = self.name_map.get(name) { |         if let Some(id) = self.name_map.get(name) { | ||||||
|             return Ok(*id) |             return Ok(*id) | ||||||
|         } |         } | ||||||
| @@ -32,22 +24,15 @@ impl FieldsMap { | |||||||
|         Ok(id) |         Ok(id) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn remove(&mut self, name: &str) { |     pub(crate) fn id(&self, name: &str) -> Option<FieldId> { | ||||||
|         if let Some(id) = self.name_map.get(name) { |  | ||||||
|             self.id_map.remove(&id); |  | ||||||
|         } |  | ||||||
|         self.name_map.remove(name); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn id(&self, name: &str) -> Option<FieldId> { |  | ||||||
|         self.name_map.get(name).copied() |         self.name_map.get(name).copied() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn name<I: Into<FieldId>>(&self, id: I) -> Option<&str> { |     pub(crate) fn name<I: Into<FieldId>>(&self, id: I) -> Option<&str> { | ||||||
|         self.id_map.get(&id.into()).map(|s| s.as_str()) |         self.id_map.get(&id.into()).map(|s| s.as_str()) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn iter(&self) -> Iter<'_, String, FieldId> { |     pub(crate) fn iter(&self) -> Iter<'_, String, FieldId> { | ||||||
|         self.name_map.iter() |         self.name_map.iter() | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -69,14 +54,10 @@ mod tests { | |||||||
|         assert_eq!(fields_map.id("title"), Some(1.into())); |         assert_eq!(fields_map.id("title"), Some(1.into())); | ||||||
|         assert_eq!(fields_map.id("descritpion"), Some(2.into())); |         assert_eq!(fields_map.id("descritpion"), Some(2.into())); | ||||||
|         assert_eq!(fields_map.id("date"), None); |         assert_eq!(fields_map.id("date"), None); | ||||||
|         assert_eq!(fields_map.len(), 3); |  | ||||||
|         assert_eq!(fields_map.name(0), Some("id")); |         assert_eq!(fields_map.name(0), Some("id")); | ||||||
|         assert_eq!(fields_map.name(1), Some("title")); |         assert_eq!(fields_map.name(1), Some("title")); | ||||||
|         assert_eq!(fields_map.name(2), Some("descritpion")); |         assert_eq!(fields_map.name(2), Some("descritpion")); | ||||||
|         assert_eq!(fields_map.name(4), None); |         assert_eq!(fields_map.name(4), None); | ||||||
|         fields_map.remove("title"); |         assert_eq!(fields_map.insert("title").unwrap(), 1.into()); | ||||||
|         assert_eq!(fields_map.id("title"), None); |  | ||||||
|         assert_eq!(fields_map.insert("title").unwrap(), 3.into()); |  | ||||||
|         assert_eq!(fields_map.len(), 3); |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,9 +1,10 @@ | |||||||
| mod error; | mod error; | ||||||
| mod fields_map; | mod fields_map; | ||||||
| mod schema; | mod schema; | ||||||
|  | mod position_map; | ||||||
|  |  | ||||||
| pub use error::{Error, SResult}; | pub use error::{Error, SResult}; | ||||||
| pub use fields_map::FieldsMap; | use fields_map::FieldsMap; | ||||||
| pub use schema::Schema; | pub use schema::Schema; | ||||||
| use serde::{Deserialize, Serialize}; | use serde::{Deserialize, Serialize}; | ||||||
| use zerocopy::{AsBytes, FromBytes}; | use zerocopy::{AsBytes, FromBytes}; | ||||||
|   | |||||||
							
								
								
									
										161
									
								
								meilisearch-schema/src/position_map.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										161
									
								
								meilisearch-schema/src/position_map.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,161 @@ | |||||||
|  | use std::collections::BTreeMap; | ||||||
|  |  | ||||||
|  | use crate::{FieldId, IndexedPos}; | ||||||
|  | use serde::{Deserialize, Serialize}; | ||||||
|  |  | ||||||
|  | #[derive(Debug, Clone, Serialize, Deserialize, Default)] | ||||||
|  | pub struct PositionMap { | ||||||
|  |     pos_to_field: Vec<FieldId>, | ||||||
|  |     field_to_pos: BTreeMap<FieldId, IndexedPos>, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl PositionMap { | ||||||
|  |     /// insert `id` at the specified `position` updating the other position if a shit if caused by | ||||||
|  |     /// the operation. If `id` is already present in the position map, it is moved to the requested | ||||||
|  |     /// `position`, potentially causing shifts. | ||||||
|  |     pub fn insert(&mut self, id: FieldId, position: IndexedPos) -> IndexedPos { | ||||||
|  |         let mut upos = position.0 as usize; | ||||||
|  |         let mut must_rebuild_map = false; | ||||||
|  |  | ||||||
|  |         if let Some(old_pos) = self.field_to_pos.get(&id) { | ||||||
|  |             let uold_pos = old_pos.0 as usize; | ||||||
|  |             self.pos_to_field.remove(uold_pos); | ||||||
|  |             must_rebuild_map = true; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if upos < self.pos_to_field.len() { | ||||||
|  |             self.pos_to_field.insert(upos, id); | ||||||
|  |             must_rebuild_map = true; | ||||||
|  |         } else { | ||||||
|  |             upos = self.pos_to_field.len(); | ||||||
|  |             self.pos_to_field.push(id); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // we only need to update all the positions if there have been a shift a some point. In | ||||||
|  |         // most cases we only did a push, so we don't need to rebuild the `field_to_pos` map. | ||||||
|  |         if must_rebuild_map { | ||||||
|  |             self.field_to_pos.clear(); | ||||||
|  |             self.field_to_pos.extend( | ||||||
|  |                 self.pos_to_field | ||||||
|  |                 .iter() | ||||||
|  |                 .enumerate() | ||||||
|  |                 .map(|(p, f)| (*f, IndexedPos(p as u16))), | ||||||
|  |             ); | ||||||
|  |         } else { | ||||||
|  |             self.field_to_pos.insert(id, IndexedPos(upos as u16)); | ||||||
|  |         } | ||||||
|  |         IndexedPos(upos as u16) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// Pushes `id` in last position | ||||||
|  |     pub fn push(&mut self, id: FieldId) -> IndexedPos { | ||||||
|  |         let pos = self.len(); | ||||||
|  |         self.insert(id, IndexedPos(pos as u16)) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pub fn len(&self) -> usize { | ||||||
|  |         self.pos_to_field.len() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pub fn field_to_pos(&self, id: FieldId) -> Option<IndexedPos> { | ||||||
|  |         self.field_to_pos.get(&id).cloned() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pub fn pos_to_field(&self, pos: IndexedPos) -> Option<FieldId> { | ||||||
|  |         let pos = pos.0 as usize; | ||||||
|  |         self.pos_to_field.get(pos).cloned() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pub fn field_pos(&self) -> impl Iterator<Item = (FieldId, IndexedPos)> + '_ { | ||||||
|  |         self.pos_to_field | ||||||
|  |             .iter() | ||||||
|  |             .enumerate() | ||||||
|  |             .map(|(i, f)| (*f, IndexedPos(i as u16))) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[cfg(test)] | ||||||
|  | mod test { | ||||||
|  |     use super::*; | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_default() { | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", PositionMap::default()), | ||||||
|  |             r##"PositionMap { pos_to_field: [], field_to_pos: {} }"## | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_insert() { | ||||||
|  |         let mut map = PositionMap::default(); | ||||||
|  |         // changing position removes from old position | ||||||
|  |         map.insert(0.into(), 0.into()); | ||||||
|  |         map.insert(1.into(), 1.into()); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", map), | ||||||
|  |             r##"PositionMap { pos_to_field: [FieldId(0), FieldId(1)], field_to_pos: {FieldId(0): IndexedPos(0), FieldId(1): IndexedPos(1)} }"## | ||||||
|  |         ); | ||||||
|  |         map.insert(0.into(), 1.into()); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", map), | ||||||
|  |             r##"PositionMap { pos_to_field: [FieldId(1), FieldId(0)], field_to_pos: {FieldId(0): IndexedPos(1), FieldId(1): IndexedPos(0)} }"## | ||||||
|  |         ); | ||||||
|  |         map.insert(2.into(), 1.into()); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", map), | ||||||
|  |             r##"PositionMap { pos_to_field: [FieldId(1), FieldId(2), FieldId(0)], field_to_pos: {FieldId(0): IndexedPos(2), FieldId(1): IndexedPos(0), FieldId(2): IndexedPos(1)} }"## | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_push() { | ||||||
|  |         let mut map = PositionMap::default(); | ||||||
|  |         map.push(0.into()); | ||||||
|  |         map.push(2.into()); | ||||||
|  |         assert_eq!(map.len(), 2); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", map), | ||||||
|  |             r##"PositionMap { pos_to_field: [FieldId(0), FieldId(2)], field_to_pos: {FieldId(0): IndexedPos(0), FieldId(2): IndexedPos(1)} }"## | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_field_to_pos() { | ||||||
|  |         let mut map = PositionMap::default(); | ||||||
|  |         map.push(0.into()); | ||||||
|  |         map.push(2.into()); | ||||||
|  |         assert_eq!(map.field_to_pos(2.into()), Some(1.into())); | ||||||
|  |         assert_eq!(map.field_to_pos(0.into()), Some(0.into())); | ||||||
|  |         assert_eq!(map.field_to_pos(4.into()), None); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_pos_to_field() { | ||||||
|  |         let mut map = PositionMap::default(); | ||||||
|  |         map.push(0.into()); | ||||||
|  |         map.push(2.into()); | ||||||
|  |         map.push(3.into()); | ||||||
|  |         map.push(4.into()); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", map), | ||||||
|  |             r##"PositionMap { pos_to_field: [FieldId(0), FieldId(2), FieldId(3), FieldId(4)], field_to_pos: {FieldId(0): IndexedPos(0), FieldId(2): IndexedPos(1), FieldId(3): IndexedPos(2), FieldId(4): IndexedPos(3)} }"## | ||||||
|  |         ); | ||||||
|  |         assert_eq!(map.pos_to_field(0.into()), Some(0.into())); | ||||||
|  |         assert_eq!(map.pos_to_field(1.into()), Some(2.into())); | ||||||
|  |         assert_eq!(map.pos_to_field(2.into()), Some(3.into())); | ||||||
|  |         assert_eq!(map.pos_to_field(3.into()), Some(4.into())); | ||||||
|  |         assert_eq!(map.pos_to_field(4.into()), None); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_field_pos() { | ||||||
|  |         let mut map = PositionMap::default(); | ||||||
|  |         map.push(0.into()); | ||||||
|  |         map.push(2.into()); | ||||||
|  |         let mut iter = map.field_pos(); | ||||||
|  |         assert_eq!(iter.next(), Some((0.into(), 0.into()))); | ||||||
|  |         assert_eq!(iter.next(), Some((2.into(), 1.into()))); | ||||||
|  |         assert_eq!(iter.next(), None); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,39 +1,10 @@ | |||||||
| use crate::{FieldsMap, FieldId, SResult, Error, IndexedPos}; |  | ||||||
| use serde::{Serialize, Deserialize}; |  | ||||||
| use std::collections::{HashMap, HashSet}; |  | ||||||
| use std::borrow::Cow; | use std::borrow::Cow; | ||||||
|  | use std::collections::{BTreeSet, HashSet}; | ||||||
|  |  | ||||||
| #[derive(Clone, Debug, Serialize, Deserialize)] | use serde::{Deserialize, Serialize}; | ||||||
| enum OptionAll<T> { |  | ||||||
|     All, |  | ||||||
|     Some(T), |  | ||||||
|     None, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| impl<T> OptionAll<T> { | use crate::position_map::PositionMap; | ||||||
|     // replace the value with None and return the previous value | use crate::{Error, FieldId, FieldsMap, IndexedPos, SResult}; | ||||||
|     fn take(&mut self) -> OptionAll<T> { |  | ||||||
|         std::mem::replace(self, OptionAll::None) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fn map<U, F: FnOnce(T) -> U>(self, f: F) -> OptionAll<U> { |  | ||||||
|         match self { |  | ||||||
|             OptionAll::Some(x) => OptionAll::Some(f(x)), |  | ||||||
|             OptionAll::All => OptionAll::All, |  | ||||||
|             OptionAll::None => OptionAll::None, |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn is_all(&self) -> bool { |  | ||||||
|         matches!(self, OptionAll::All) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| impl<T> Default for OptionAll<T> { |  | ||||||
|     fn default() -> OptionAll<T> { |  | ||||||
|         OptionAll::All |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[derive(Clone, Debug, Serialize, Deserialize, Default)] | #[derive(Clone, Debug, Serialize, Deserialize, Default)] | ||||||
| pub struct Schema { | pub struct Schema { | ||||||
| @@ -41,34 +12,26 @@ pub struct Schema { | |||||||
|  |  | ||||||
|     primary_key: Option<FieldId>, |     primary_key: Option<FieldId>, | ||||||
|     ranked: HashSet<FieldId>, |     ranked: HashSet<FieldId>, | ||||||
|     displayed: OptionAll<HashSet<FieldId>>, |     displayed: Option<BTreeSet<FieldId>>, | ||||||
|  |  | ||||||
|     indexed: OptionAll<Vec<FieldId>>, |     searchable: Option<Vec<FieldId>>, | ||||||
|     indexed_map: HashMap<FieldId, IndexedPos>, |     pub indexed_position: PositionMap, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl Schema { | impl Schema { | ||||||
|     pub fn new() -> Schema { |  | ||||||
|         Schema::default() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn with_primary_key(name: &str) -> Schema { |     pub fn with_primary_key(name: &str) -> Schema { | ||||||
|         let mut fields_map = FieldsMap::default(); |         let mut fields_map = FieldsMap::default(); | ||||||
|         let field_id = fields_map.insert(name).unwrap(); |         let field_id = fields_map.insert(name).unwrap(); | ||||||
|  |         let mut indexed_position = PositionMap::default(); | ||||||
|         let mut displayed = HashSet::new(); |         indexed_position.push(field_id); | ||||||
|         let mut indexed_map = HashMap::new(); |  | ||||||
|  |  | ||||||
|         displayed.insert(field_id); |  | ||||||
|         indexed_map.insert(field_id, 0.into()); |  | ||||||
|  |  | ||||||
|         Schema { |         Schema { | ||||||
|             fields_map, |             fields_map, | ||||||
|             primary_key: Some(field_id), |             primary_key: Some(field_id), | ||||||
|             ranked: HashSet::new(), |             ranked: HashSet::new(), | ||||||
|             displayed: OptionAll::All, |             displayed: None, | ||||||
|             indexed: OptionAll::All, |             searchable: None, | ||||||
|             indexed_map, |             indexed_position, | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -78,13 +41,11 @@ impl Schema { | |||||||
|  |  | ||||||
|     pub fn set_primary_key(&mut self, name: &str) -> SResult<FieldId> { |     pub fn set_primary_key(&mut self, name: &str) -> SResult<FieldId> { | ||||||
|         if self.primary_key.is_some() { |         if self.primary_key.is_some() { | ||||||
|             return Err(Error::PrimaryKeyAlreadyPresent) |             return Err(Error::PrimaryKeyAlreadyPresent); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         let id = self.insert(name)?; |         let id = self.insert(name)?; | ||||||
|         self.primary_key = Some(id); |         self.primary_key = Some(id); | ||||||
|         self.set_indexed(name)?; |  | ||||||
|         self.set_displayed(name)?; |  | ||||||
|  |  | ||||||
|         Ok(id) |         Ok(id) | ||||||
|     } |     } | ||||||
| @@ -101,202 +62,98 @@ impl Schema { | |||||||
|         self.fields_map.iter().map(|(k, _)| k.as_ref()) |         self.fields_map.iter().map(|(k, _)| k.as_ref()) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn contains(&self, name: &str) -> bool { |     /// add `name` to the list of known fields | ||||||
|         self.fields_map.id(name).is_some() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn insert(&mut self, name: &str) -> SResult<FieldId> { |     pub fn insert(&mut self, name: &str) -> SResult<FieldId> { | ||||||
|         self.fields_map.insert(name) |         self.fields_map.insert(name) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn insert_and_index(&mut self, name: &str) -> SResult<FieldId> { |     /// Adds `name` to the list of known fields, and in the last position of the indexed_position map. This | ||||||
|         match self.fields_map.id(name) { |     /// field is taken into acccount when `searchableAttribute` or `displayedAttributes` is set to `"*"` | ||||||
|             Some(id) => { |     pub fn insert_with_position(&mut self, name: &str) -> SResult<(FieldId, IndexedPos)> { | ||||||
|                 Ok(id) |         let field_id = self.fields_map.insert(name)?; | ||||||
|             } |         let position = self | ||||||
|             None => { |             .is_searchable(field_id) | ||||||
|                 self.set_indexed(name)?; |             .unwrap_or_else(|| self.indexed_position.push(field_id)); | ||||||
|                 self.set_displayed(name) |         Ok((field_id, position)) | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn ranked(&self) -> &HashSet<FieldId> { |     pub fn ranked(&self) -> &HashSet<FieldId> { | ||||||
|         &self.ranked |         &self.ranked | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn ranked_name(&self) -> HashSet<&str> { |     fn displayed(&self) -> Cow<BTreeSet<FieldId>> { | ||||||
|         self.ranked.iter().filter_map(|a| self.name(*a)).collect() |         match &self.displayed { | ||||||
|     } |             Some(displayed) => Cow::Borrowed(displayed), | ||||||
|  |             None => Cow::Owned(self.indexed_position.field_pos().map(|(f, _)| f).collect()), | ||||||
|     pub fn displayed(&self) -> Cow<HashSet<FieldId>> { |  | ||||||
|         match self.displayed { |  | ||||||
|             OptionAll::Some(ref v) => Cow::Borrowed(v), |  | ||||||
|             OptionAll::All => { |  | ||||||
|                 let fields = self |  | ||||||
|                     .fields_map |  | ||||||
|                     .iter() |  | ||||||
|                     .map(|(_, &v)| v) |  | ||||||
|                     .collect::<HashSet<_>>(); |  | ||||||
|                 Cow::Owned(fields) |  | ||||||
|             } |  | ||||||
|             OptionAll::None => Cow::Owned(HashSet::new()) |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn is_displayed_all(&self) -> bool { |     pub fn is_displayed_all(&self) -> bool { | ||||||
|         self.displayed.is_all() |         self.displayed.is_none() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn displayed_name(&self) -> HashSet<&str> { |     pub fn displayed_names(&self) -> BTreeSet<&str> { | ||||||
|         match self.displayed { |         self.displayed() | ||||||
|             OptionAll::All => self.fields_map.iter().filter_map(|(_, &v)| self.name(v)).collect(), |  | ||||||
|             OptionAll::Some(ref v) => v.iter().filter_map(|a| self.name(*a)).collect(), |  | ||||||
|             OptionAll::None => HashSet::new(), |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn indexed(&self) -> Cow<[FieldId]> { |  | ||||||
|         match self.indexed { |  | ||||||
|             OptionAll::Some(ref v) => Cow::Borrowed(v), |  | ||||||
|             OptionAll::All => { |  | ||||||
|                 let fields = self |  | ||||||
|                     .fields_map |  | ||||||
|             .iter() |             .iter() | ||||||
|                     .map(|(_, &f)| f) |             .filter_map(|&f| self.name(f)) | ||||||
|                     .collect(); |             .collect() | ||||||
|                 Cow::Owned(fields) |     } | ||||||
|             }, |  | ||||||
|             OptionAll::None => Cow::Owned(Vec::new()) |     fn searchable(&self) -> Cow<[FieldId]> { | ||||||
|  |         match &self.searchable { | ||||||
|  |             Some(searchable) => Cow::Borrowed(&searchable), | ||||||
|  |             None => Cow::Owned(self.indexed_position.field_pos().map(|(f, _)| f).collect()), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn indexed_name(&self) -> Vec<&str> { |     pub fn searchable_names(&self) -> Vec<&str> { | ||||||
|         self.indexed().iter().filter_map(|a| self.name(*a)).collect() |         self.searchable() | ||||||
|  |             .iter() | ||||||
|  |             .filter_map(|a| self.name(*a)) | ||||||
|  |             .collect() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn set_ranked(&mut self, name: &str) -> SResult<FieldId> { |     pub(crate) fn set_ranked(&mut self, name: &str) -> SResult<FieldId> { | ||||||
|         let id = self.fields_map.insert(name)?; |         let id = self.fields_map.insert(name)?; | ||||||
|         self.ranked.insert(id); |         self.ranked.insert(id); | ||||||
|         Ok(id) |         Ok(id) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn set_displayed(&mut self, name: &str) -> SResult<FieldId> { |  | ||||||
|         let id = self.fields_map.insert(name)?; |  | ||||||
|         self.displayed = match self.displayed.take() { |  | ||||||
|             OptionAll::All => OptionAll::All, |  | ||||||
|             OptionAll::None => { |  | ||||||
|                 let mut displayed = HashSet::new(); |  | ||||||
|                 displayed.insert(id); |  | ||||||
|                 OptionAll::Some(displayed) |  | ||||||
|             }, |  | ||||||
|             OptionAll::Some(mut v) => { |  | ||||||
|                 v.insert(id); |  | ||||||
|                 OptionAll::Some(v) |  | ||||||
|             } |  | ||||||
|         }; |  | ||||||
|         Ok(id) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn set_indexed(&mut self, name: &str) -> SResult<(FieldId, IndexedPos)> { |  | ||||||
|         let id = self.fields_map.insert(name)?; |  | ||||||
|  |  | ||||||
|         if let Some(indexed_pos) = self.indexed_map.get(&id) { |  | ||||||
|             return Ok((id, *indexed_pos)) |  | ||||||
|         }; |  | ||||||
|         let pos = self.indexed_map.len() as u16; |  | ||||||
|         self.indexed_map.insert(id, pos.into()); |  | ||||||
|         self.indexed = self.indexed.take().map(|mut v| { |  | ||||||
|             v.push(id); |  | ||||||
|             v |  | ||||||
|         }); |  | ||||||
|         Ok((id, pos.into())) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn clear_ranked(&mut self) { |     pub fn clear_ranked(&mut self) { | ||||||
|         self.ranked.clear(); |         self.ranked.clear(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn remove_ranked(&mut self, name: &str) { |  | ||||||
|         if let Some(id) = self.fields_map.id(name) { |  | ||||||
|             self.ranked.remove(&id); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /// remove field from displayed attributes. If diplayed attributes is OptionAll::All, |  | ||||||
|     /// dipslayed attributes is turned into OptionAll::Some(v) where v is all displayed attributes |  | ||||||
|     /// except name. |  | ||||||
|     pub fn remove_displayed(&mut self, name: &str) { |  | ||||||
|         if let Some(id) = self.fields_map.id(name) { |  | ||||||
|             self.displayed = match self.displayed.take() { |  | ||||||
|                 OptionAll::Some(mut v) => { |  | ||||||
|                     v.remove(&id); |  | ||||||
|                     OptionAll::Some(v) |  | ||||||
|                 } |  | ||||||
|                 OptionAll::All => { |  | ||||||
|                     let displayed = self.fields_map |  | ||||||
|                         .iter() |  | ||||||
|                         .filter_map(|(key, &value)| { |  | ||||||
|                             if key != name { |  | ||||||
|                                 Some(value) |  | ||||||
|                             } else { |  | ||||||
|                                 None |  | ||||||
|                             } |  | ||||||
|                         }) |  | ||||||
|                         .collect::<HashSet<_>>(); |  | ||||||
|                     OptionAll::Some(displayed) |  | ||||||
|                 } |  | ||||||
|                 OptionAll::None => OptionAll::None, |  | ||||||
|             }; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn remove_indexed(&mut self, name: &str) { |  | ||||||
|         if let Some(id) = self.fields_map.id(name) { |  | ||||||
|             self.indexed_map.remove(&id); |  | ||||||
|             self.indexed = match self.indexed.take() { |  | ||||||
|                 // valid because indexed is All and indexed() return the content of |  | ||||||
|                 // indexed_map that is already updated |  | ||||||
|                 OptionAll::All => OptionAll::Some(self.indexed().into_owned()), |  | ||||||
|                 OptionAll::Some(mut v) => { |  | ||||||
|                     v.retain(|x| *x != id); |  | ||||||
|                     OptionAll::Some(v) |  | ||||||
|                 } |  | ||||||
|                 OptionAll::None => OptionAll::None, |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn is_ranked(&self, id: FieldId) -> bool { |     pub fn is_ranked(&self, id: FieldId) -> bool { | ||||||
|         self.ranked.get(&id).is_some() |         self.ranked.get(&id).is_some() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn is_displayed(&self, id: FieldId) -> bool { |     pub fn is_displayed(&self, id: FieldId) -> bool { | ||||||
|         match self.displayed { |         match &self.displayed { | ||||||
|             OptionAll::Some(ref v) => v.contains(&id), |             Some(displayed) => displayed.contains(&id), | ||||||
|             OptionAll::All => true, |             None => true, | ||||||
|             OptionAll::None => false, |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn is_indexed(&self, id: FieldId) -> Option<&IndexedPos> { |     pub fn is_searchable(&self, id: FieldId) -> Option<IndexedPos> { | ||||||
|         self.indexed_map.get(&id) |         match &self.searchable { | ||||||
|  |             Some(searchable) if searchable.contains(&id) => self.indexed_position.field_to_pos(id), | ||||||
|  |             None => self.indexed_position.field_to_pos(id), | ||||||
|  |             _ => None, | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn is_indexed_all(&self) -> bool { |     pub fn is_searchable_all(&self) -> bool { | ||||||
|         self.indexed.is_all() |         self.searchable.is_none() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn indexed_pos_to_field_id<I: Into<IndexedPos>>(&self, pos: I) -> Option<FieldId> { |     pub fn indexed_pos_to_field_id<I: Into<IndexedPos>>(&self, pos: I) -> Option<FieldId> { | ||||||
|         let indexed_pos = pos.into().0; |         self.indexed_position.pos_to_field(pos.into()) | ||||||
|         self |  | ||||||
|             .indexed_map |  | ||||||
|             .iter() |  | ||||||
|             .find(|(_, &v)| v.0 == indexed_pos) |  | ||||||
|             .map(|(&k, _)| k) |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn update_ranked<S: AsRef<str>>(&mut self, data: impl IntoIterator<Item = S>) -> SResult<()> { |     pub fn update_ranked<S: AsRef<str>>( | ||||||
|  |         &mut self, | ||||||
|  |         data: impl IntoIterator<Item = S>, | ||||||
|  |     ) -> SResult<()> { | ||||||
|         self.ranked.clear(); |         self.ranked.clear(); | ||||||
|         for name in data { |         for name in data { | ||||||
|             self.set_ranked(name.as_ref())?; |             self.set_ranked(name.as_ref())?; | ||||||
| @@ -304,46 +161,211 @@ impl Schema { | |||||||
|         Ok(()) |         Ok(()) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn update_displayed<S: AsRef<str>>(&mut self, data: impl IntoIterator<Item = S>) -> SResult<()> { |     pub fn update_displayed<S: AsRef<str>>( | ||||||
|         self.displayed = match self.displayed.take() { |         &mut self, | ||||||
|             OptionAll::Some(mut v) => { |         data: impl IntoIterator<Item = S>, | ||||||
|                 v.clear(); |     ) -> SResult<()> { | ||||||
|                 OptionAll::Some(v) |         let mut displayed = BTreeSet::new(); | ||||||
|             } |  | ||||||
|             _ => OptionAll::Some(HashSet::new()) |  | ||||||
|         }; |  | ||||||
|         for name in data { |         for name in data { | ||||||
|             self.set_displayed(name.as_ref())?; |             let id = self.fields_map.insert(name.as_ref())?; | ||||||
|  |             displayed.insert(id); | ||||||
|         } |         } | ||||||
|  |         self.displayed.replace(displayed); | ||||||
|         Ok(()) |         Ok(()) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn update_indexed<S: AsRef<str>>(&mut self, data: Vec<S>) -> SResult<()> { |     pub fn update_searchable<S: AsRef<str>>(&mut self, data: Vec<S>) -> SResult<()> { | ||||||
|         self.indexed = match self.indexed.take() { |         let mut searchable = Vec::with_capacity(data.len()); | ||||||
|             OptionAll::Some(mut v) => { |         for (pos, name) in data.iter().enumerate() { | ||||||
|                 v.clear(); |             let id = self.insert(name.as_ref())?; | ||||||
|                 OptionAll::Some(v) |             self.indexed_position.insert(id, IndexedPos(pos as u16)); | ||||||
|             }, |             searchable.push(id); | ||||||
|             _ => OptionAll::Some(Vec::new()), |  | ||||||
|         }; |  | ||||||
|         self.indexed_map.clear(); |  | ||||||
|         for name in data { |  | ||||||
|             self.set_indexed(name.as_ref())?; |  | ||||||
|         } |         } | ||||||
|  |         self.searchable.replace(searchable); | ||||||
|         Ok(()) |         Ok(()) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn set_all_fields_as_indexed(&mut self) { |     pub fn set_all_searchable(&mut self) { | ||||||
|         self.indexed = OptionAll::All; |         self.searchable.take(); | ||||||
|         self.indexed_map.clear(); |  | ||||||
|  |  | ||||||
|         for (_name, id) in self.fields_map.iter() { |  | ||||||
|             let pos = self.indexed_map.len() as u16; |  | ||||||
|             self.indexed_map.insert(*id, pos.into()); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn set_all_fields_as_displayed(&mut self) { |     pub fn set_all_displayed(&mut self) { | ||||||
|         self.displayed = OptionAll::All |         self.displayed.take(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[cfg(test)] | ||||||
|  | mod test { | ||||||
|  |     use super::*; | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_with_primary_key() { | ||||||
|  |         let schema = Schema::with_primary_key("test"); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", schema), | ||||||
|  |             r##"Schema { fields_map: FieldsMap { name_map: {"test": FieldId(0)}, id_map: {FieldId(0): "test"}, next_id: FieldId(1) }, primary_key: Some(FieldId(0)), ranked: {}, displayed: None, searchable: None, indexed_position: PositionMap { pos_to_field: [FieldId(0)], field_to_pos: {FieldId(0): IndexedPos(0)} } }"## | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn primary_key() { | ||||||
|  |         let schema = Schema::with_primary_key("test"); | ||||||
|  |         assert_eq!(schema.primary_key(), Some("test")); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_insert_with_position_base() { | ||||||
|  |         let mut schema = Schema::default(); | ||||||
|  |         let (id, position) = schema.insert_with_position("foo").unwrap(); | ||||||
|  |         assert!(schema.searchable.is_none()); | ||||||
|  |         assert!(schema.displayed.is_none()); | ||||||
|  |         assert_eq!(id, 0.into()); | ||||||
|  |         assert_eq!(position, 0.into()); | ||||||
|  |         let (id, position) = schema.insert_with_position("bar").unwrap(); | ||||||
|  |         assert_eq!(id, 1.into()); | ||||||
|  |         assert_eq!(position, 1.into()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_insert_with_position_primary_key() { | ||||||
|  |         let mut schema = Schema::with_primary_key("test"); | ||||||
|  |         let (id, position) = schema.insert_with_position("foo").unwrap(); | ||||||
|  |         assert!(schema.searchable.is_none()); | ||||||
|  |         assert!(schema.displayed.is_none()); | ||||||
|  |         assert_eq!(id, 1.into()); | ||||||
|  |         assert_eq!(position, 1.into()); | ||||||
|  |         let (id, position) = schema.insert_with_position("test").unwrap(); | ||||||
|  |         assert_eq!(id, 0.into()); | ||||||
|  |         assert_eq!(position, 0.into()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_insert_with_position_non_all_searchable_attributes() {} | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_insert() { | ||||||
|  |         let mut schema = Schema::default(); | ||||||
|  |         let field_id = schema.insert("foo").unwrap(); | ||||||
|  |         assert!(schema.fields_map.name(field_id).is_some()); | ||||||
|  |         assert!(schema.searchable.is_none()); | ||||||
|  |         assert!(schema.displayed.is_none()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_update_searchable() { | ||||||
|  |         let mut schema = Schema::default(); | ||||||
|  |  | ||||||
|  |         schema.update_searchable(vec!["foo", "bar"]).unwrap(); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", schema.indexed_position), | ||||||
|  |             r##"PositionMap { pos_to_field: [FieldId(0), FieldId(1)], field_to_pos: {FieldId(0): IndexedPos(0), FieldId(1): IndexedPos(1)} }"## | ||||||
|  |         ); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", schema.searchable), | ||||||
|  |             r##"Some([FieldId(0), FieldId(1)])"## | ||||||
|  |         ); | ||||||
|  |         schema.update_searchable(vec!["bar"]).unwrap(); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", schema.searchable), | ||||||
|  |             r##"Some([FieldId(1)])"## | ||||||
|  |         ); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", schema.indexed_position), | ||||||
|  |             r##"PositionMap { pos_to_field: [FieldId(1), FieldId(0)], field_to_pos: {FieldId(0): IndexedPos(1), FieldId(1): IndexedPos(0)} }"## | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_update_displayed() { | ||||||
|  |         let mut schema = Schema::default(); | ||||||
|  |         schema.update_displayed(vec!["foobar"]).unwrap(); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", schema.displayed), | ||||||
|  |             r##"Some({FieldId(0)})"## | ||||||
|  |         ); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", schema.indexed_position), | ||||||
|  |             r##"PositionMap { pos_to_field: [], field_to_pos: {} }"## | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_is_searchable_all() { | ||||||
|  |         let mut schema = Schema::default(); | ||||||
|  |         assert!(schema.is_searchable_all()); | ||||||
|  |         schema.update_searchable(vec!["foo"]).unwrap(); | ||||||
|  |         assert!(!schema.is_searchable_all()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_is_displayed_all() { | ||||||
|  |         let mut schema = Schema::default(); | ||||||
|  |         assert!(schema.is_displayed_all()); | ||||||
|  |         schema.update_displayed(vec!["foo"]).unwrap(); | ||||||
|  |         assert!(!schema.is_displayed_all()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_searchable_names() { | ||||||
|  |         let mut schema = Schema::default(); | ||||||
|  |         assert_eq!(format!("{:?}", schema.searchable_names()), r##"[]"##); | ||||||
|  |         schema.insert_with_position("foo").unwrap(); | ||||||
|  |         schema.insert_with_position("bar").unwrap(); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", schema.searchable_names()), | ||||||
|  |             r##"["foo", "bar"]"## | ||||||
|  |         ); | ||||||
|  |         schema.update_searchable(vec!["hello", "world"]).unwrap(); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", schema.searchable_names()), | ||||||
|  |             r##"["hello", "world"]"## | ||||||
|  |         ); | ||||||
|  |         schema.set_all_searchable(); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", schema.searchable_names()), | ||||||
|  |             r##"["hello", "world", "foo", "bar"]"## | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_displayed_names() { | ||||||
|  |         let mut schema = Schema::default(); | ||||||
|  |         assert_eq!(format!("{:?}", schema.displayed_names()), r##"{}"##); | ||||||
|  |         schema.insert_with_position("foo").unwrap(); | ||||||
|  |         schema.insert_with_position("bar").unwrap(); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", schema.displayed_names()), | ||||||
|  |             r##"{"bar", "foo"}"## | ||||||
|  |         ); | ||||||
|  |         schema.update_displayed(vec!["hello", "world"]).unwrap(); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", schema.displayed_names()), | ||||||
|  |             r##"{"hello", "world"}"## | ||||||
|  |         ); | ||||||
|  |         schema.set_all_displayed(); | ||||||
|  |         assert_eq!( | ||||||
|  |             format!("{:?}", schema.displayed_names()), | ||||||
|  |             r##"{"bar", "foo"}"## | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_set_all_searchable() { | ||||||
|  |         let mut schema = Schema::default(); | ||||||
|  |         assert!(schema.is_searchable_all()); | ||||||
|  |         schema.update_searchable(vec!["foobar"]).unwrap(); | ||||||
|  |         assert!(!schema.is_searchable_all()); | ||||||
|  |         schema.set_all_searchable(); | ||||||
|  |         assert!(schema.is_searchable_all()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_set_all_displayed() { | ||||||
|  |         let mut schema = Schema::default(); | ||||||
|  |         assert!(schema.is_displayed_all()); | ||||||
|  |         schema.update_displayed(vec!["foobar"]).unwrap(); | ||||||
|  |         assert!(!schema.is_displayed_all()); | ||||||
|  |         schema.set_all_displayed(); | ||||||
|  |         assert!(schema.is_displayed_all()); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user