use super::{expression::*, operation::*};
use crate::{filter, IntoBson};
use itertools::Itertools;
use mongodb::bson::{doc, Bson};
pub(crate) trait IntoUpdateExpression {
fn into_update_expression(self) -> crate::Result<UpdateExpression>;
}
pub(crate) trait IntoUpdateExpressions {
fn into_update_expressions(self) -> crate::Result<Vec<UpdateExpression>>;
}
impl IntoUpdateExpressions for UpdateOperation {
fn into_update_expressions(self) -> crate::Result<Vec<UpdateExpression>> {
match self {
UpdateOperation::Generic(generic) => Ok(vec![generic.into_update_expression()?]),
UpdateOperation::UpdateMany(update_many) => Ok(vec![update_many.into_update_expression()?]),
UpdateOperation::Unset(unset) => Ok(vec![unset.into_update_expression()?]),
UpdateOperation::Upsert(upsert) => upsert.into_update_expressions(),
}
}
}
impl IntoUpdateExpression for GenericOperation {
fn into_update_expression(self) -> crate::Result<UpdateExpression> {
Ok(UpdateExpression::set(self.field_path, self.expression))
}
}
impl IntoUpdateExpressions for Upsert {
fn into_update_expressions(self) -> crate::Result<Vec<UpdateExpression>> {
let should_set_id = format!("__prisma_should_set__{}", &self.field_path.identifier());
let should_set_ref_id = format!("${}", &should_set_id);
let mut expressions: Vec<UpdateExpression> = vec![];
expressions.push(
doc! { "$addFields": { &should_set_id: Upsert::render_should_set_condition(self.field_path()) }}.into(),
);
expressions.push(
Set::from(self.set)
.into_conditional_set(doc! { "$eq": [&should_set_ref_id, true] })
.into(),
);
for op in self.updates {
match op {
UpdateOperation::Upsert(upsert) => {
expressions.extend(upsert.into_update_expressions()?);
}
operation => {
let exprs = operation
.into_update_expressions()?
.into_iter()
.map(|expr| {
let set = expr
.try_into_set()
.expect("all upsert's update expressions should be `Set`s")
.into_conditional_set(doc! { "$eq": [&should_set_ref_id, false] });
UpdateExpression::from(set)
})
.collect_vec();
expressions.extend(exprs);
}
}
}
expressions.push(doc! { "$unset": Bson::String(should_set_id) }.into());
Ok(expressions)
}
}
impl IntoUpdateExpression for UpdateMany {
fn into_update_expression(self) -> crate::Result<UpdateExpression> {
let field_path = self.field_path.clone();
let elem_alias = self.elem_alias.clone();
let ref_elem_alias = format!("$${}", &elem_alias);
let (filter_doc, _) = filter::MongoFilterVisitor::new(format!("${}", &elem_alias), false)
.visit(self.filter.clone())?
.render();
let merge_objects = self.into_merge_objects_expr()?;
let map_expr = doc! {
"$map": {
"input": field_path.dollar_path(true),
"as": &elem_alias,
"in": {
"$cond": {
"if": filter_doc,
"then": merge_objects.into_bson()?,
"else": Bson::String(ref_elem_alias)
}
}
}
};
Ok(UpdateExpression::set(field_path, map_expr))
}
}
impl IntoUpdateExpression for Unset {
fn into_update_expression(self) -> crate::Result<UpdateExpression> {
Ok(UpdateExpression::set(self.field_path, Bson::from("$$REMOVE")))
}
}