2020-10-22 14:23:33 +02:00
|
|
|
use std::collections::BTreeMap;
|
2020-10-22 12:50:04 +02:00
|
|
|
use serde::{Serialize, Deserialize};
|
2020-11-26 17:38:08 +01:00
|
|
|
use crate::FieldId;
|
2020-10-22 12:50:04 +02:00
|
|
|
|
|
|
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
|
|
|
pub struct FieldsIdsMap {
|
2020-11-26 17:38:08 +01:00
|
|
|
names_ids: BTreeMap<String, FieldId>,
|
|
|
|
ids_names: BTreeMap<FieldId, String>,
|
|
|
|
next_id: Option<FieldId>,
|
2020-10-22 12:50:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl FieldsIdsMap {
|
|
|
|
pub fn new() -> FieldsIdsMap {
|
|
|
|
FieldsIdsMap {
|
2020-10-22 14:23:33 +02:00
|
|
|
names_ids: BTreeMap::new(),
|
2020-10-22 12:50:04 +02:00
|
|
|
ids_names: BTreeMap::new(),
|
|
|
|
next_id: Some(0),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the number of fields ids in the map.
|
|
|
|
pub fn len(&self) -> usize {
|
|
|
|
self.names_ids.len()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns `true` if the map is empty.
|
|
|
|
pub fn is_empty(&self) -> bool {
|
|
|
|
self.names_ids.is_empty()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the field id related to a field name, it will create a new field id if the
|
|
|
|
/// name is not already known. Returns `None` if the maximum field id as been reached.
|
2020-11-26 17:38:08 +01:00
|
|
|
pub fn insert(&mut self, name: &str) -> Option<FieldId> {
|
2020-10-22 12:50:04 +02:00
|
|
|
match self.names_ids.get(name) {
|
|
|
|
Some(id) => Some(*id),
|
|
|
|
None => {
|
|
|
|
let id = self.next_id?;
|
|
|
|
self.next_id = id.checked_add(1);
|
|
|
|
self.names_ids.insert(name.to_owned(), id);
|
|
|
|
self.ids_names.insert(id, name.to_owned());
|
|
|
|
Some(id)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the id of a field based on its name.
|
2020-11-26 17:38:08 +01:00
|
|
|
pub fn id(&self, name: &str) -> Option<FieldId> {
|
2020-10-22 12:50:04 +02:00
|
|
|
self.names_ids.get(name).copied()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the name of a field based on its id.
|
2020-11-26 17:38:08 +01:00
|
|
|
pub fn name(&self, id: FieldId) -> Option<&str> {
|
2020-10-22 12:50:04 +02:00
|
|
|
self.ids_names.get(&id).map(String::as_str)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Remove a field name and id based on its name.
|
2020-11-26 17:38:08 +01:00
|
|
|
pub fn remove(&mut self, name: &str) -> Option<FieldId> {
|
2020-10-22 12:50:04 +02:00
|
|
|
match self.names_ids.remove(name) {
|
|
|
|
Some(id) => self.ids_names.remove_entry(&id).map(|(id, _)| id),
|
|
|
|
None => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Iterate over the ids and names in the ids order.
|
2020-11-26 17:38:08 +01:00
|
|
|
pub fn iter(&self) -> impl Iterator<Item=(FieldId, &str)> {
|
2020-10-22 12:50:04 +02:00
|
|
|
self.ids_names.iter().map(|(id, name)| (*id, name.as_str()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-22 14:23:33 +02:00
|
|
|
impl Default for FieldsIdsMap {
|
|
|
|
fn default() -> FieldsIdsMap {
|
|
|
|
FieldsIdsMap::new()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-22 12:50:04 +02:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn fields_ids_map() {
|
|
|
|
let mut map = FieldsIdsMap::new();
|
|
|
|
|
|
|
|
assert_eq!(map.insert("id"), Some(0));
|
|
|
|
assert_eq!(map.insert("title"), Some(1));
|
|
|
|
assert_eq!(map.insert("description"), Some(2));
|
|
|
|
assert_eq!(map.insert("id"), Some(0));
|
|
|
|
assert_eq!(map.insert("title"), Some(1));
|
|
|
|
assert_eq!(map.insert("description"), Some(2));
|
|
|
|
|
|
|
|
assert_eq!(map.id("id"), Some(0));
|
|
|
|
assert_eq!(map.id("title"), Some(1));
|
|
|
|
assert_eq!(map.id("description"), Some(2));
|
|
|
|
assert_eq!(map.id("date"), None);
|
|
|
|
|
|
|
|
assert_eq!(map.len(), 3);
|
|
|
|
|
|
|
|
assert_eq!(map.name(0), Some("id"));
|
|
|
|
assert_eq!(map.name(1), Some("title"));
|
|
|
|
assert_eq!(map.name(2), Some("description"));
|
|
|
|
assert_eq!(map.name(4), None);
|
|
|
|
|
|
|
|
assert_eq!(map.remove("title"), Some(1));
|
|
|
|
|
|
|
|
assert_eq!(map.id("title"), None);
|
|
|
|
assert_eq!(map.insert("title"), Some(3));
|
|
|
|
assert_eq!(map.len(), 3);
|
|
|
|
|
|
|
|
let mut iter = map.iter();
|
|
|
|
assert_eq!(iter.next(), Some((0, "id")));
|
|
|
|
assert_eq!(iter.next(), Some((2, "description")));
|
|
|
|
assert_eq!(iter.next(), Some((3, "title")));
|
|
|
|
assert_eq!(iter.next(), None);
|
|
|
|
}
|
|
|
|
}
|