diesel/query_builder/upsert/
on_conflict_actions.rs

1use std::marker::PhantomData;
2
3use crate::backend::sql_dialect::on_conflict_clause;
4use crate::expression::{AppearsOnTable, Expression};
5use crate::query_builder::*;
6use crate::query_source::*;
7
8#[doc(hidden)]
9#[derive(Debug, Clone, Copy, QueryId)]
10pub struct DoNothing<T>(PhantomData<T>);
11
12impl<T> DoNothing<T> {
13    pub(crate) fn new() -> Self {
14        Self(PhantomData)
15    }
16}
17
18impl<T, DB> QueryFragment<DB> for DoNothing<T>
19where
20    DB: Backend,
21    Self: QueryFragment<DB, DB::OnConflictClause>,
22{
23    fn walk_ast<'b>(&'b self, pass: AstPass<'_, 'b, DB>) -> QueryResult<()> {
24        <Self as QueryFragment<DB, DB::OnConflictClause>>::walk_ast(self, pass)
25    }
26}
27
28impl<DB, T, SD> QueryFragment<DB, SD> for DoNothing<T>
29where
30    DB: Backend<OnConflictClause = SD>,
31    SD: on_conflict_clause::PgLikeOnConflictClause,
32{
33    fn walk_ast<'b>(&'b self, mut out: AstPass<'_, 'b, DB>) -> QueryResult<()> {
34        out.push_sql(" DO NOTHING");
35        Ok(())
36    }
37}
38
39#[doc(hidden)]
40#[derive(Debug, Clone, Copy, QueryId)]
41pub struct DoUpdate<T, Tab> {
42    pub(crate) changeset: T,
43    tab: PhantomData<Tab>,
44}
45
46impl<T, Tab> DoUpdate<T, Tab> {
47    pub(crate) fn new(changeset: T) -> Self {
48        DoUpdate {
49            changeset,
50            tab: PhantomData,
51        }
52    }
53}
54
55impl<DB, T, Tab> QueryFragment<DB> for DoUpdate<T, Tab>
56where
57    DB: Backend,
58    Self: QueryFragment<DB, DB::OnConflictClause>,
59{
60    fn walk_ast<'b>(&'b self, pass: AstPass<'_, 'b, DB>) -> QueryResult<()> {
61        <Self as QueryFragment<DB, DB::OnConflictClause>>::walk_ast(self, pass)
62    }
63}
64
65impl<DB, T, Tab, SD> QueryFragment<DB, SD> for DoUpdate<T, Tab>
66where
67    DB: Backend<OnConflictClause = SD>,
68    T: QueryFragment<DB>,
69    SD: on_conflict_clause::PgLikeOnConflictClause,
70{
71    fn walk_ast<'b>(&'b self, mut out: AstPass<'_, 'b, DB>) -> QueryResult<()> {
72        out.unsafe_to_cache_prepared();
73        if self.changeset.is_noop(out.backend())? {
74            out.push_sql(" DO NOTHING");
75        } else {
76            out.push_sql(" DO UPDATE SET ");
77            self.changeset.walk_ast(out.reborrow())?;
78        }
79        Ok(())
80    }
81}
82
83#[doc(hidden)]
84#[derive(Debug, Clone, Copy, QueryId)]
85pub struct Excluded<T>(T);
86
87impl<T> Excluded<T> {
88    pub(crate) fn new(t: T) -> Self {
89        Excluded(t)
90    }
91}
92
93impl<DB, T> QueryFragment<DB> for Excluded<T>
94where
95    DB: Backend,
96    Self: QueryFragment<DB, DB::OnConflictClause>,
97{
98    fn walk_ast<'b>(&'b self, pass: AstPass<'_, 'b, DB>) -> QueryResult<()> {
99        <Self as QueryFragment<DB, DB::OnConflictClause>>::walk_ast(self, pass)
100    }
101}
102
103impl<DB, T, SD> QueryFragment<DB, SD> for Excluded<T>
104where
105    DB: Backend<OnConflictClause = SD>,
106    T: Column,
107    SD: on_conflict_clause::PgLikeOnConflictClause,
108{
109    fn walk_ast<'b>(&'b self, mut out: AstPass<'_, 'b, DB>) -> QueryResult<()> {
110        out.push_sql("excluded.");
111        out.push_identifier(T::NAME)?;
112        Ok(())
113    }
114}
115
116impl<T> Expression for Excluded<T>
117where
118    T: Expression,
119{
120    type SqlType = T::SqlType;
121}
122
123impl<T> AppearsOnTable<T::Table> for Excluded<T>
124where
125    T: Column,
126    Excluded<T>: Expression,
127{
128}