pub(crate) mod changeset;
pub(super) mod target;
use crate::backend::DieselReserveSpecialization;
use crate::dsl::{Filter, IntoBoxed};
use crate::expression::{
is_aggregate, AppearsOnTable, Expression, MixedAggregates, SelectableExpression, ValidGrouping,
};
use crate::query_builder::returning_clause::*;
use crate::query_builder::where_clause::*;
use crate::query_dsl::methods::{BoxedDsl, FilterDsl};
use crate::query_dsl::RunQueryDsl;
use crate::query_source::Table;
use crate::result::EmptyChangeset;
use crate::result::Error::QueryBuilderError;
use crate::{query_builder::*, QuerySource};
pub(crate) use self::private::UpdateAutoTypeHelper;
impl<T: QuerySource, U> UpdateStatement<T, U, SetNotCalled> {
pub(crate) fn new(target: UpdateTarget<T, U>) -> Self {
UpdateStatement {
from_clause: target.table.from_clause(),
where_clause: target.where_clause,
values: SetNotCalled,
returning: NoReturningClause,
}
}
pub fn set<V>(self, values: V) -> UpdateStatement<T, U, V::Changeset>
where
T: Table,
V: changeset::AsChangeset<Target = T>,
UpdateStatement<T, U, V::Changeset>: AsQuery,
{
UpdateStatement {
from_clause: self.from_clause,
where_clause: self.where_clause,
values: values.as_changeset(),
returning: self.returning,
}
}
}
#[derive(Clone, Debug)]
#[must_use = "Queries are only executed when calling `load`, `get_result` or similar."]
pub struct UpdateStatement<T: QuerySource, U, V = SetNotCalled, Ret = NoReturningClause> {
from_clause: T::FromClause,
where_clause: U,
values: V,
returning: Ret,
}
pub type BoxedUpdateStatement<'a, DB, T, V = SetNotCalled, Ret = NoReturningClause> =
UpdateStatement<T, BoxedWhereClause<'a, DB>, V, Ret>;
impl<T: QuerySource, U, V, Ret> UpdateStatement<T, U, V, Ret> {
pub fn filter<Predicate>(self, predicate: Predicate) -> Filter<Self, Predicate>
where
Self: FilterDsl<Predicate>,
{
FilterDsl::filter(self, predicate)
}
pub fn into_boxed<'a, DB>(self) -> IntoBoxed<'a, Self, DB>
where
DB: Backend,
Self: BoxedDsl<'a, DB>,
{
BoxedDsl::internal_into_boxed(self)
}
}
impl<T, U, V, Ret, Predicate> FilterDsl<Predicate> for UpdateStatement<T, U, V, Ret>
where
T: QuerySource,
U: WhereAnd<Predicate>,
Predicate: AppearsOnTable<T>,
{
type Output = UpdateStatement<T, U::Output, V, Ret>;
fn filter(self, predicate: Predicate) -> Self::Output {
UpdateStatement {
from_clause: self.from_clause,
where_clause: self.where_clause.and(predicate),
values: self.values,
returning: self.returning,
}
}
}
impl<'a, T, U, V, Ret, DB> BoxedDsl<'a, DB> for UpdateStatement<T, U, V, Ret>
where
T: QuerySource,
U: Into<BoxedWhereClause<'a, DB>>,
{
type Output = BoxedUpdateStatement<'a, DB, T, V, Ret>;
fn internal_into_boxed(self) -> Self::Output {
UpdateStatement {
from_clause: self.from_clause,
where_clause: self.where_clause.into(),
values: self.values,
returning: self.returning,
}
}
}
impl<T, U, V, Ret, DB> QueryFragment<DB> for UpdateStatement<T, U, V, Ret>
where
DB: Backend + DieselReserveSpecialization,
T: Table,
T::FromClause: QueryFragment<DB>,
U: QueryFragment<DB>,
V: QueryFragment<DB>,
Ret: QueryFragment<DB>,
{
fn walk_ast<'b>(&'b self, mut out: AstPass<'_, 'b, DB>) -> QueryResult<()> {
if self.values.is_noop(out.backend())? {
return Err(QueryBuilderError(Box::new(EmptyChangeset)));
}
out.unsafe_to_cache_prepared();
out.push_sql("UPDATE ");
self.from_clause.walk_ast(out.reborrow())?;
out.push_sql(" SET ");
self.values.walk_ast(out.reborrow())?;
self.where_clause.walk_ast(out.reborrow())?;
self.returning.walk_ast(out.reborrow())?;
Ok(())
}
}
impl<T, U, V, Ret> QueryId for UpdateStatement<T, U, V, Ret>
where
T: QuerySource,
{
type QueryId = ();
const HAS_STATIC_QUERY_ID: bool = false;
}
impl<T, U, V> AsQuery for UpdateStatement<T, U, V, NoReturningClause>
where
T: Table,
UpdateStatement<T, U, V, ReturningClause<T::AllColumns>>: Query,
T::AllColumns: ValidGrouping<()>,
<T::AllColumns as ValidGrouping<()>>::IsAggregate:
MixedAggregates<is_aggregate::No, Output = is_aggregate::No>,
{
type SqlType = <Self::Query as Query>::SqlType;
type Query = UpdateStatement<T, U, V, ReturningClause<T::AllColumns>>;
fn as_query(self) -> Self::Query {
self.returning(T::all_columns())
}
}
impl<T, U, V, Ret> Query for UpdateStatement<T, U, V, ReturningClause<Ret>>
where
T: Table,
Ret: Expression + SelectableExpression<T> + ValidGrouping<()>,
Ret::IsAggregate: MixedAggregates<is_aggregate::No, Output = is_aggregate::No>,
{
type SqlType = Ret::SqlType;
}
impl<T: QuerySource, U, V, Ret, Conn> RunQueryDsl<Conn> for UpdateStatement<T, U, V, Ret> {}
impl<T: QuerySource, U, V> UpdateStatement<T, U, V, NoReturningClause> {
pub fn returning<E>(self, returns: E) -> UpdateStatement<T, U, V, ReturningClause<E>>
where
T: Table,
UpdateStatement<T, U, V, ReturningClause<E>>: Query,
{
UpdateStatement {
from_clause: self.from_clause,
where_clause: self.where_clause,
values: self.values,
returning: ReturningClause(returns),
}
}
}
#[derive(Debug, Clone, Copy)]
pub struct SetNotCalled;
mod private {
#[allow(unreachable_pub)]
pub trait UpdateAutoTypeHelper {
type Table;
type Where;
}
impl<T, W> UpdateAutoTypeHelper for crate::query_builder::UpdateStatement<T, W>
where
T: crate::QuerySource,
{
type Table = T;
type Where = W;
}
}