use mongodb::bson::Bson;
use serde::{Deserialize, Serialize};
use std::{
collections::BTreeMap,
fmt::{self, Debug},
ops,
};
use crate::{CollectionWalker, IndexWalker};
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash, Serialize, Deserialize)]
pub struct CollectionId(usize);
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash, Serialize, Deserialize)]
pub struct IndexId(usize);
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct CollectionData {
pub(crate) name: String,
pub(crate) has_schema: bool,
pub(crate) is_capped: bool,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize)]
pub enum IndexType {
Normal,
Unique,
Fulltext,
}
impl IndexType {
pub fn is_fulltext(self) -> bool {
matches!(self, Self::Fulltext)
}
}
impl IndexData {
pub fn is_fulltext(&self) -> bool {
self.r#type.is_fulltext()
}
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct IndexData {
pub name: String,
pub r#type: IndexType,
pub fields: Vec<IndexField>,
pub collection_id: CollectionId,
}
#[derive(Debug, Default, Clone, Serialize, Deserialize)]
pub struct MongoSchema {
collections: Vec<CollectionData>,
indexes: Vec<IndexData>,
pub(super) collection_indexes: BTreeMap<CollectionId, Vec<IndexId>>,
}
impl MongoSchema {
pub fn push_collection(&mut self, name: String, has_schema: bool, is_capped: bool) -> CollectionId {
self.collections.push(CollectionData {
name,
has_schema,
is_capped,
});
CollectionId(self.collections.len() - 1)
}
pub fn push_index(
&mut self,
collection_id: CollectionId,
name: String,
r#type: IndexType,
fields: Vec<IndexField>,
) -> IndexId {
self.indexes.push(IndexData {
name,
r#type,
fields,
collection_id,
});
let index_id = IndexId(self.indexes.len() - 1);
let coll_ind = self.collection_indexes.entry(collection_id).or_default();
coll_ind.push(index_id);
index_id
}
pub fn walk_collections(&self) -> impl ExactSizeIterator<Item = CollectionWalker<'_>> + '_ {
self.collections.iter().enumerate().map(|(id, _)| CollectionWalker {
id: CollectionId(id),
schema: self,
})
}
pub fn walk_collection(&self, id: CollectionId) -> CollectionWalker<'_> {
CollectionWalker { id, schema: self }
}
pub fn walk_index(&self, id: IndexId) -> IndexWalker<'_> {
IndexWalker { id, schema: self }
}
pub fn remove_fulltext_indexes(&mut self) {
self.collection_indexes.clear();
#[allow(clippy::needless_collect)] let indexes: Vec<_> = self.indexes.drain(0..).filter(|i| !i.is_fulltext()).collect();
for index in indexes.into_iter() {
let IndexData {
name,
r#type,
fields,
collection_id,
} = index;
self.push_index(collection_id, name, r#type, fields);
}
}
}
impl ops::Index<CollectionId> for MongoSchema {
type Output = CollectionData;
fn index(&self, index: CollectionId) -> &Self::Output {
&self.collections[index.0]
}
}
impl ops::Index<IndexId> for MongoSchema {
type Output = IndexData;
fn index(&self, index: IndexId) -> &Self::Output {
&self.indexes[index.0]
}
}
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
pub struct IndexField {
pub name: String,
pub property: IndexFieldProperty,
}
impl IndexField {
pub fn name(&self) -> &str {
&self.name
}
pub fn is_text(&self) -> bool {
matches!(self.property, IndexFieldProperty::Text)
}
}
impl fmt::Display for IndexField {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "\"{}\":{}", self.name, self.property)
}
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize)]
pub enum IndexFieldProperty {
Text,
Ascending,
Descending,
}
impl IndexFieldProperty {
pub fn is_descending(self) -> bool {
matches!(self, Self::Descending)
}
}
impl fmt::Display for IndexFieldProperty {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
IndexFieldProperty::Text => f.write_str("\"text\""),
IndexFieldProperty::Ascending => f.write_str("1"),
IndexFieldProperty::Descending => f.write_str("-1"),
}
}
}
impl From<IndexFieldProperty> for Bson {
fn from(property: IndexFieldProperty) -> Self {
match property {
IndexFieldProperty::Text => Bson::String(String::from("text")),
IndexFieldProperty::Ascending => Bson::Int32(1),
IndexFieldProperty::Descending => Bson::Int32(-1),
}
}
}