use super::{into_expression::IntoUpdateExpression, operation};
use connector_interface::FieldPath;
use indexmap::IndexMap;
use mongodb::bson::{doc, Bson, Document};
#[derive(Debug, Clone)]
pub(crate) enum UpdateExpression {
Set(Set),
IfThenElse(IfThenElse),
MergeObjects(MergeObjects),
MergeDocument(MergeDocument),
Generic(Bson),
}
impl UpdateExpression {
pub fn set(field_path: FieldPath, expression: impl Into<UpdateExpression>) -> Self {
Self::Set(Set {
field_path,
expression: Box::new(expression.into()),
})
}
pub fn if_then_else(
cond: impl Into<UpdateExpression>,
then: impl Into<UpdateExpression>,
els: impl Into<UpdateExpression>,
) -> Self {
Self::IfThenElse(IfThenElse {
cond: Box::new(cond.into()),
then: Box::new(then.into()),
els: Box::new(els.into()),
})
}
pub(crate) fn try_into_set(self) -> Option<Set> {
if let Self::Set(v) = self {
Some(v)
} else {
None
}
}
pub(crate) fn as_merge_objects_mut(&mut self) -> Option<&mut MergeObjects> {
if let Self::MergeObjects(ref mut v) = self {
Some(v)
} else {
None
}
}
}
#[derive(Debug, Clone)]
pub(crate) struct Set {
pub field_path: FieldPath,
pub expression: Box<UpdateExpression>,
}
impl Set {
pub(crate) fn field_path(&self) -> &FieldPath {
&self.field_path
}
pub(crate) fn into_conditional_set(self, cond: impl Into<UpdateExpression>) -> Self {
let dollar_path = self.field_path().dollar_path(true);
Self {
field_path: self.field_path,
expression: Box::new(UpdateExpression::IfThenElse(IfThenElse {
cond: Box::new(cond.into()),
then: self.expression,
els: Box::new(UpdateExpression::Generic(Bson::String(dollar_path))),
})),
}
}
}
#[derive(Debug, Clone)]
pub(crate) struct IfThenElse {
pub cond: Box<UpdateExpression>,
pub then: Box<UpdateExpression>,
pub els: Box<UpdateExpression>,
}
#[derive(Debug, Clone, Default)]
pub(crate) struct MergeDocument {
pub inner: IndexMap<String, UpdateExpression>,
}
impl IntoIterator for MergeDocument {
type Item = (String, UpdateExpression);
type IntoIter = indexmap::map::IntoIter<String, UpdateExpression>;
fn into_iter(self) -> Self::IntoIter {
self.inner.into_iter()
}
}
impl MergeDocument {
pub(crate) fn inner_mut(&mut self) -> &mut IndexMap<String, UpdateExpression> {
&mut self.inner
}
}
#[derive(Debug, Clone)]
pub(crate) struct MergeObjects {
pub(crate) field_path: FieldPath,
pub(crate) document: MergeDocument,
pub(crate) keys_to_unset: Vec<String>,
}
impl MergeObjects {
pub(crate) fn new(field_path: FieldPath) -> Self {
Self {
field_path,
document: MergeDocument::default(),
keys_to_unset: vec![],
}
}
pub(crate) fn field_path(&self) -> &FieldPath {
&self.field_path
}
pub(crate) fn doc_mut(&mut self) -> &mut IndexMap<String, UpdateExpression> {
self.document.inner_mut()
}
pub(crate) fn insert_set(&mut self, set: Set) {
let set_expr = set.expression.clone();
let (target, inner_merge) = self.fill_nested_merge_objects(set.field_path());
inner_merge.doc_mut().insert(target.to_owned(), *set_expr);
}
pub(crate) fn insert_upsert(&mut self, upsert: operation::Upsert) -> crate::Result<()> {
let field_path = upsert.field_path().clone();
let mut merge_doc = MergeObjects::new(field_path.clone());
let (target, inner_merge) = self.fill_nested_merge_objects(&field_path);
for op in upsert.updates {
match op {
operation::UpdateOperation::Generic(generic) => {
let mut set = generic.into_update_expression()?.try_into_set().unwrap();
set.field_path.keep_last();
merge_doc.insert_set(set);
}
operation::UpdateOperation::UpdateMany(update_many) => {
let mut set = update_many.into_update_expression()?.try_into_set().unwrap();
set.field_path.keep_last();
merge_doc.insert_set(set);
}
operation::UpdateOperation::Upsert(mut upsert) => {
upsert.field_path.keep_last();
merge_doc.insert_upsert(upsert)?;
}
operation::UpdateOperation::Unset(mut unset) => {
unset.field_path.keep_last();
merge_doc.insert_unset(unset);
}
}
}
let new_if = UpdateExpression::if_then_else(
doc! { "$eq": [operation::Upsert::render_should_set_condition(&upsert.set.field_path().clone()), true] },
upsert.set.expression,
merge_doc,
);
inner_merge.doc_mut().insert(target.to_owned(), new_if);
Ok(())
}
pub(crate) fn insert_unset(&mut self, unset: operation::Unset) {
let (target, inner_merge) = self.fill_nested_merge_objects(unset.field_path());
inner_merge.keys_to_unset.push(target.clone());
}
fn fill_nested_merge_objects<'a, 'b>(
&'a mut self,
field_path: &'b FieldPath,
) -> (&'b String, &'a mut MergeObjects) {
let (target, segments) = field_path.path.split_last().unwrap();
let inner_merge = segments.iter().enumerate().fold(self, |acc, (depth, segment)| {
let mut merge_path = field_path.clone();
merge_path.take(depth + 1);
let inner_merge = acc
.doc_mut()
.entry(segment.to_string())
.or_insert(MergeObjects::new(merge_path).into())
.as_merge_objects_mut()
.unwrap();
inner_merge
});
(target, inner_merge)
}
}
impl From<Bson> for UpdateExpression {
fn from(bson: Bson) -> Self {
Self::Generic(bson)
}
}
impl From<Document> for UpdateExpression {
fn from(doc: Document) -> Self {
Self::Generic(doc.into())
}
}
impl From<Set> for UpdateExpression {
fn from(set: Set) -> Self {
Self::Set(set)
}
}
impl From<IfThenElse> for UpdateExpression {
fn from(if_then_else: IfThenElse) -> Self {
Self::IfThenElse(if_then_else)
}
}
impl From<MergeDocument> for UpdateExpression {
fn from(merge_doc: MergeDocument) -> Self {
Self::MergeDocument(merge_doc)
}
}
impl From<MergeObjects> for UpdateExpression {
fn from(merge_objects: MergeObjects) -> Self {
Self::MergeObjects(merge_objects)
}
}