diesel/expression/functions/aggregate_expressions/
frame_clause.rs

1use crate::backend::sql_dialect;
2use crate::query_builder::{QueryFragment, QueryId};
3
4use super::aggregate_filter::NoFilter;
5use super::aggregate_order::{NoOrder, Order};
6use super::over_clause::{NoWindow, OverClause, ValidAggregateFilterForWindow};
7use super::partition_by::NoPartition;
8use super::prefix::NoPrefix;
9use super::{AggregateExpression, IsWindowFunction};
10
11pub struct NoFrame;
#[automatically_derived]
impl ::core::fmt::Debug for NoFrame {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f, "NoFrame")
    }
}
#[automatically_derived]
impl ::core::clone::Clone for NoFrame {
    #[inline]
    fn clone(&self) -> NoFrame { *self }
}
#[automatically_derived]
impl ::core::marker::Copy for NoFrame { }
const _: () =
    {
        use diesel;
        #[allow(non_camel_case_types)]
        impl diesel::query_builder::QueryId for NoFrame {
            type QueryId = NoFrame<>;
            const HAS_STATIC_QUERY_ID: bool = true;
            const IS_WINDOW_FUNCTION: bool = false;
        }
    };
impl<DB> crate::query_builder::QueryFragment<DB> for NoFrame where
    DB: crate::backend::Backend + crate::backend::DieselReserveSpecialization
    {
    fn walk_ast<'b>(&'b self,
        _pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        Ok(())
    }
}empty_clause!(NoFrame);
12
13#[derive(const _: () =
    {
        use diesel;
        #[allow(non_camel_case_types)]
        impl<F: diesel::query_builder::QueryId> diesel::query_builder::QueryId
            for FrameClause<F> {
            type QueryId =
                FrameClause<<F as diesel::query_builder::QueryId>::QueryId>;
            const HAS_STATIC_QUERY_ID: bool =
                <F as diesel::query_builder::QueryId>::HAS_STATIC_QUERY_ID &&
                    true;
            const IS_WINDOW_FUNCTION: bool =
                <F as diesel::query_builder::QueryId>::IS_WINDOW_FUNCTION ||
                    false;
        }
    };QueryId, #[automatically_derived]
impl<F: ::core::marker::Copy> ::core::marker::Copy for FrameClause<F> { }Copy, #[automatically_derived]
impl<F: ::core::clone::Clone> ::core::clone::Clone for FrameClause<F> {
    #[inline]
    fn clone(&self) -> FrameClause<F> {
        FrameClause(::core::clone::Clone::clone(&self.0))
    }
}Clone, #[automatically_derived]
impl<F: ::core::fmt::Debug> ::core::fmt::Debug for FrameClause<F> {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_tuple_field1_finish(f, "FrameClause",
            &&self.0)
    }
}Debug)]
14pub struct FrameClause<F>(F);
15
16impl<F, DB> QueryFragment<DB> for FrameClause<F>
17where
18    F: QueryFragment<DB>,
19    DB: crate::backend::Backend,
20{
21    fn walk_ast<'b>(
22        &'b self,
23        pass: crate::query_builder::AstPass<'_, 'b, DB>,
24    ) -> crate::QueryResult<()> {
25        self.0.walk_ast(pass)?;
26        Ok(())
27    }
28}
29
30macro_rules! simple_frame_expr {
31    ($(#[doc = $doc: literal])* $name: ident, $kind: expr) => {
32        #[derive(QueryId, Clone, Copy, Debug)]
33        $(#[doc = $doc])*
34        pub struct $name;
35
36        impl<DB> QueryFragment<DB> for $name
37        where
38            DB: crate::backend::Backend,
39        {
40            fn walk_ast<'b>(
41                &'b self,
42                mut pass: crate::query_builder::AstPass<'_, 'b, DB>,
43            ) -> crate::QueryResult<()> {
44                pass.push_sql($kind);
45                Ok(())
46            }
47        }
48    };
49}
50
51macro_rules! simple_frame_expr_with_bound {
52    ($(#[doc = $doc:literal])* $name: ident, $option: ident, $bound: ty, $kind: expr) => {
53        #[derive(QueryId, Clone, Copy, Debug)]
54        $(#[doc = $doc])*
55        pub struct $name;
56
57        impl<DB> QueryFragment<DB> for $name
58        where
59            DB: crate::backend::Backend,
60            Self: QueryFragment<DB, DB::$option>,
61        {
62            fn walk_ast<'b>(
63                &'b self,
64                pass: crate::query_builder::AstPass<'_, 'b, DB>,
65            ) -> crate::QueryResult<()> {
66                <Self as QueryFragment<DB, DB::$option>>::walk_ast(self, pass)
67            }
68        }
69        impl<DB> QueryFragment<DB, $bound> for $name
70        where
71            DB: crate::backend::Backend<$option = $bound>,
72        {
73            fn walk_ast<'b>(
74                &'b self,
75                mut pass: crate::query_builder::AstPass<'_, 'b, DB>,
76            ) -> crate::QueryResult<()> {
77                pass.push_sql($kind);
78                Ok(())
79            }
80        }
81    };
82}
83
84// kinds
85#[doc = r" Range frame mode"]
#[doc = r""]
#[doc = r" Requires to set a `ORDER BY` clause set via `window_order`"]
#[doc = r" for the current window function call"]
pub struct Range;
const _: () =
    {
        use diesel;
        #[allow(non_camel_case_types)]
        impl diesel::query_builder::QueryId for Range {
            type QueryId = Range<>;
            const HAS_STATIC_QUERY_ID: bool = true;
            const IS_WINDOW_FUNCTION: bool = false;
        }
    };
#[automatically_derived]
impl ::core::clone::Clone for Range {
    #[inline]
    fn clone(&self) -> Range { *self }
}
#[automatically_derived]
impl ::core::marker::Copy for Range { }
#[automatically_derived]
impl ::core::fmt::Debug for Range {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f, "Range")
    }
}
impl<DB> QueryFragment<DB> for Range where DB: crate::backend::Backend {
    fn walk_ast<'b>(&'b self,
        mut pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        pass.push_sql(" RANGE ");
        Ok(())
    }
}simple_frame_expr!(
86    /// Range frame mode
87    ///
88    /// Requires to set a `ORDER BY` clause set via `window_order`
89    /// for the current window function call
90    Range, " RANGE ");
91#[doc = r" Rows frame mode"]
#[doc = r""]
#[doc = r" This options specifies that the offset starts/ends"]
#[doc = r" a certain number of rows before/after the current row"]
pub struct Rows;
const _: () =
    {
        use diesel;
        #[allow(non_camel_case_types)]
        impl diesel::query_builder::QueryId for Rows {
            type QueryId = Rows<>;
            const HAS_STATIC_QUERY_ID: bool = true;
            const IS_WINDOW_FUNCTION: bool = false;
        }
    };
#[automatically_derived]
impl ::core::clone::Clone for Rows {
    #[inline]
    fn clone(&self) -> Rows { *self }
}
#[automatically_derived]
impl ::core::marker::Copy for Rows { }
#[automatically_derived]
impl ::core::fmt::Debug for Rows {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f, "Rows")
    }
}
impl<DB> QueryFragment<DB> for Rows where DB: crate::backend::Backend {
    fn walk_ast<'b>(&'b self,
        mut pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        pass.push_sql(" ROWS ");
        Ok(())
    }
}simple_frame_expr!(
92    /// Rows frame mode
93    ///
94    /// This options specifies that the offset starts/ends
95    /// a certain number of rows before/after the current row
96    Rows, " ROWS ");
97#[doc = r" Groups frame mode"]
#[doc = r""]
#[doc = r" This options specifies that the offset starts/ends"]
#[doc = r" a certain number of peer groups before or after"]
#[doc = r" the current row. A peer group is a set of rows that are"]
#[doc = r" equivalent in the `ORDER BY` ordering"]
#[doc = r""]
#[doc = r" There must be a `ORDER BY` clause set via `window_order`"]
#[doc = r" for the current window function call"]
pub struct Groups;
const _: () =
    {
        use diesel;
        #[allow(non_camel_case_types)]
        impl diesel::query_builder::QueryId for Groups {
            type QueryId = Groups<>;
            const HAS_STATIC_QUERY_ID: bool = true;
            const IS_WINDOW_FUNCTION: bool = false;
        }
    };
#[automatically_derived]
impl ::core::clone::Clone for Groups {
    #[inline]
    fn clone(&self) -> Groups { *self }
}
#[automatically_derived]
impl ::core::marker::Copy for Groups { }
#[automatically_derived]
impl ::core::fmt::Debug for Groups {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f, "Groups")
    }
}
impl<DB> QueryFragment<DB> for Groups where DB: crate::backend::Backend,
    Self: QueryFragment<DB, DB::WindowFrameClauseGroupSupport> {
    fn walk_ast<'b>(&'b self, pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        <Self as
                QueryFragment<DB,
                DB::WindowFrameClauseGroupSupport>>::walk_ast(self, pass)
    }
}
impl<DB>
    QueryFragment<DB,
    sql_dialect::window_frame_clause_group_support::IsoGroupWindowFrameUnit>
    for Groups where
    DB: crate::backend::Backend<WindowFrameClauseGroupSupport =
    sql_dialect::window_frame_clause_group_support::IsoGroupWindowFrameUnit> {
    fn walk_ast<'b>(&'b self,
        mut pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        pass.push_sql(" GROUPS ");
        Ok(())
    }
}simple_frame_expr_with_bound!(
98    /// Groups frame mode
99    ///
100    /// This options specifies that the offset starts/ends
101    /// a certain number of peer groups before or after
102    /// the current row. A peer group is a set of rows that are
103    /// equivalent in the `ORDER BY` ordering
104    ///
105    /// There must be a `ORDER BY` clause set via `window_order`
106    /// for the current window function call
107    Groups,
108    WindowFrameClauseGroupSupport,
109    sql_dialect::window_frame_clause_group_support::IsoGroupWindowFrameUnit,
110    " GROUPS "
111);
112
113// start & end
114#[doc = r" A `UNBOUNDED PRECEDING` frame bound"]
#[doc = r""]
#[doc = r" This bound specifies that the frame starts with"]
#[doc = r" the first row of the partition"]
#[doc = r""]
#[doc = r" This option can be used as frame start"]
pub struct UnboundedPreceding;
const _: () =
    {
        use diesel;
        #[allow(non_camel_case_types)]
        impl diesel::query_builder::QueryId for UnboundedPreceding {
            type QueryId = UnboundedPreceding<>;
            const HAS_STATIC_QUERY_ID: bool = true;
            const IS_WINDOW_FUNCTION: bool = false;
        }
    };
#[automatically_derived]
impl ::core::clone::Clone for UnboundedPreceding {
    #[inline]
    fn clone(&self) -> UnboundedPreceding { *self }
}
#[automatically_derived]
impl ::core::marker::Copy for UnboundedPreceding { }
#[automatically_derived]
impl ::core::fmt::Debug for UnboundedPreceding {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f, "UnboundedPreceding")
    }
}
impl<DB> QueryFragment<DB> for UnboundedPreceding where
    DB: crate::backend::Backend {
    fn walk_ast<'b>(&'b self,
        mut pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        pass.push_sql("UNBOUNDED PRECEDING ");
        Ok(())
    }
}simple_frame_expr!(
115    /// A `UNBOUNDED PRECEDING` frame bound
116    ///
117    /// This bound specifies that the frame starts with
118    /// the first row of the partition
119    ///
120    /// This option can be used as frame start
121    UnboundedPreceding, "UNBOUNDED PRECEDING ");
122#[doc = r" A `CURRENT ROW` frame bound"]
#[doc = r""]
#[doc = r" For the `RANGE` and `GROUP` frame mode this bound"]
#[doc = r" specifies that the current frame starts with the current"]
#[doc = r" rows first peer groups row. A peer group is defined as a"]
#[doc = r" set of rows with an equivalent `ORDER BY` ordering."]
#[doc = r""]
#[doc = r" For the `ROWS` mode `CURRENT ROW` simply means the current"]
#[doc = r" row"]
#[doc = r""]
#[doc = r" This option can be used as frame start and end"]
pub struct CurrentRow;
const _: () =
    {
        use diesel;
        #[allow(non_camel_case_types)]
        impl diesel::query_builder::QueryId for CurrentRow {
            type QueryId = CurrentRow<>;
            const HAS_STATIC_QUERY_ID: bool = true;
            const IS_WINDOW_FUNCTION: bool = false;
        }
    };
#[automatically_derived]
impl ::core::clone::Clone for CurrentRow {
    #[inline]
    fn clone(&self) -> CurrentRow { *self }
}
#[automatically_derived]
impl ::core::marker::Copy for CurrentRow { }
#[automatically_derived]
impl ::core::fmt::Debug for CurrentRow {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f, "CurrentRow")
    }
}
impl<DB> QueryFragment<DB> for CurrentRow where DB: crate::backend::Backend {
    fn walk_ast<'b>(&'b self,
        mut pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        pass.push_sql("CURRENT ROW ");
        Ok(())
    }
}simple_frame_expr!(
123    /// A `CURRENT ROW` frame bound
124    ///
125    /// For the `RANGE` and `GROUP` frame mode this bound
126    /// specifies that the current frame starts with the current
127    /// rows first peer groups row. A peer group is defined as a
128    /// set of rows with an equivalent `ORDER BY` ordering.
129    ///
130    /// For the `ROWS` mode `CURRENT ROW` simply means the current
131    /// row
132    ///
133    /// This option can be used as frame start and end
134    CurrentRow, "CURRENT ROW ");
135#[doc = r" A `UNBOUNDED FOLLOWING` frame bound"]
#[doc = r""]
#[doc = r" This bound specifies that the frame ends with"]
#[doc = r" the last row of the partition"]
#[doc = r""]
#[doc = r" This option can be used as frame end"]
pub struct UnboundedFollowing;
const _: () =
    {
        use diesel;
        #[allow(non_camel_case_types)]
        impl diesel::query_builder::QueryId for UnboundedFollowing {
            type QueryId = UnboundedFollowing<>;
            const HAS_STATIC_QUERY_ID: bool = true;
            const IS_WINDOW_FUNCTION: bool = false;
        }
    };
#[automatically_derived]
impl ::core::clone::Clone for UnboundedFollowing {
    #[inline]
    fn clone(&self) -> UnboundedFollowing { *self }
}
#[automatically_derived]
impl ::core::marker::Copy for UnboundedFollowing { }
#[automatically_derived]
impl ::core::fmt::Debug for UnboundedFollowing {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f, "UnboundedFollowing")
    }
}
impl<DB> QueryFragment<DB> for UnboundedFollowing where
    DB: crate::backend::Backend {
    fn walk_ast<'b>(&'b self,
        mut pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        pass.push_sql("UNBOUNDED FOLLOWING ");
        Ok(())
    }
}simple_frame_expr!(
136    /// A `UNBOUNDED FOLLOWING` frame bound
137    ///
138    /// This bound specifies that the frame ends with
139    /// the last row of the partition
140    ///
141    /// This option can be used as frame end
142    UnboundedFollowing, "UNBOUNDED FOLLOWING ");
143
144// exclusion
145#[doc = r" Exclude the current row from the window frame"]
pub struct ExcludeCurrentRow;
const _: () =
    {
        use diesel;
        #[allow(non_camel_case_types)]
        impl diesel::query_builder::QueryId for ExcludeCurrentRow {
            type QueryId = ExcludeCurrentRow<>;
            const HAS_STATIC_QUERY_ID: bool = true;
            const IS_WINDOW_FUNCTION: bool = false;
        }
    };
#[automatically_derived]
impl ::core::clone::Clone for ExcludeCurrentRow {
    #[inline]
    fn clone(&self) -> ExcludeCurrentRow { *self }
}
#[automatically_derived]
impl ::core::marker::Copy for ExcludeCurrentRow { }
#[automatically_derived]
impl ::core::fmt::Debug for ExcludeCurrentRow {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f, "ExcludeCurrentRow")
    }
}
impl<DB> QueryFragment<DB> for ExcludeCurrentRow where
    DB: crate::backend::Backend,
    Self: QueryFragment<DB, DB::WindowFrameExclusionSupport> {
    fn walk_ast<'b>(&'b self, pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        <Self as
                QueryFragment<DB,
                DB::WindowFrameExclusionSupport>>::walk_ast(self, pass)
    }
}
impl<DB>
    QueryFragment<DB,
    sql_dialect::window_frame_exclusion_support::FrameExclusionSupport> for
    ExcludeCurrentRow where
    DB: crate::backend::Backend<WindowFrameExclusionSupport =
    sql_dialect::window_frame_exclusion_support::FrameExclusionSupport> {
    fn walk_ast<'b>(&'b self,
        mut pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        pass.push_sql("EXCLUDE CURRENT ROW ");
        Ok(())
    }
}simple_frame_expr_with_bound!(
146    /// Exclude the current row from the window frame
147    ExcludeCurrentRow,
148    WindowFrameExclusionSupport,
149    sql_dialect::window_frame_exclusion_support::FrameExclusionSupport,
150    "EXCLUDE CURRENT ROW "
151);
152#[doc = r" Exclude the current peer group from the window frame"]
#[doc = r""]
#[doc = r" A peer group is a set of rows with an equivalent `ORDER BY`"]
#[doc = r" ordering"]
pub struct ExcludeGroup;
const _: () =
    {
        use diesel;
        #[allow(non_camel_case_types)]
        impl diesel::query_builder::QueryId for ExcludeGroup {
            type QueryId = ExcludeGroup<>;
            const HAS_STATIC_QUERY_ID: bool = true;
            const IS_WINDOW_FUNCTION: bool = false;
        }
    };
#[automatically_derived]
impl ::core::clone::Clone for ExcludeGroup {
    #[inline]
    fn clone(&self) -> ExcludeGroup { *self }
}
#[automatically_derived]
impl ::core::marker::Copy for ExcludeGroup { }
#[automatically_derived]
impl ::core::fmt::Debug for ExcludeGroup {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f, "ExcludeGroup")
    }
}
impl<DB> QueryFragment<DB> for ExcludeGroup where DB: crate::backend::Backend,
    Self: QueryFragment<DB, DB::WindowFrameExclusionSupport> {
    fn walk_ast<'b>(&'b self, pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        <Self as
                QueryFragment<DB,
                DB::WindowFrameExclusionSupport>>::walk_ast(self, pass)
    }
}
impl<DB>
    QueryFragment<DB,
    sql_dialect::window_frame_exclusion_support::FrameExclusionSupport> for
    ExcludeGroup where
    DB: crate::backend::Backend<WindowFrameExclusionSupport =
    sql_dialect::window_frame_exclusion_support::FrameExclusionSupport> {
    fn walk_ast<'b>(&'b self,
        mut pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        pass.push_sql("EXCLUDE GROUP ");
        Ok(())
    }
}simple_frame_expr_with_bound!(
153    /// Exclude the current peer group from the window frame
154    ///
155    /// A peer group is a set of rows with an equivalent `ORDER BY`
156    /// ordering
157    ExcludeGroup,
158    WindowFrameExclusionSupport,
159    sql_dialect::window_frame_exclusion_support::FrameExclusionSupport,
160    "EXCLUDE GROUP "
161);
162#[doc = r" Exclude any peers, but not the current row from the window frame"]
#[doc = r""]
#[doc = r" This excludes any row with an equivalent `ORDER BY` ordering"]
#[doc = r" to the current row from the frame window"]
pub struct ExcludeTies;
const _: () =
    {
        use diesel;
        #[allow(non_camel_case_types)]
        impl diesel::query_builder::QueryId for ExcludeTies {
            type QueryId = ExcludeTies<>;
            const HAS_STATIC_QUERY_ID: bool = true;
            const IS_WINDOW_FUNCTION: bool = false;
        }
    };
#[automatically_derived]
impl ::core::clone::Clone for ExcludeTies {
    #[inline]
    fn clone(&self) -> ExcludeTies { *self }
}
#[automatically_derived]
impl ::core::marker::Copy for ExcludeTies { }
#[automatically_derived]
impl ::core::fmt::Debug for ExcludeTies {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f, "ExcludeTies")
    }
}
impl<DB> QueryFragment<DB> for ExcludeTies where DB: crate::backend::Backend,
    Self: QueryFragment<DB, DB::WindowFrameExclusionSupport> {
    fn walk_ast<'b>(&'b self, pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        <Self as
                QueryFragment<DB,
                DB::WindowFrameExclusionSupport>>::walk_ast(self, pass)
    }
}
impl<DB>
    QueryFragment<DB,
    sql_dialect::window_frame_exclusion_support::FrameExclusionSupport> for
    ExcludeTies where
    DB: crate::backend::Backend<WindowFrameExclusionSupport =
    sql_dialect::window_frame_exclusion_support::FrameExclusionSupport> {
    fn walk_ast<'b>(&'b self,
        mut pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        pass.push_sql("EXCLUDE TIES ");
        Ok(())
    }
}simple_frame_expr_with_bound!(
163    /// Exclude any peers, but not the current row from the window frame
164    ///
165    /// This excludes any row with an equivalent `ORDER BY` ordering
166    /// to the current row from the frame window
167    ExcludeTies,
168    WindowFrameExclusionSupport,
169    sql_dialect::window_frame_exclusion_support::FrameExclusionSupport,
170    "EXCLUDE TIES "
171);
172#[doc = r" Exclude no rows from the frame windows"]
#[doc = r""]
#[doc = r" This is the default behaviour if not specified"]
pub struct ExcludeNoOthers;
const _: () =
    {
        use diesel;
        #[allow(non_camel_case_types)]
        impl diesel::query_builder::QueryId for ExcludeNoOthers {
            type QueryId = ExcludeNoOthers<>;
            const HAS_STATIC_QUERY_ID: bool = true;
            const IS_WINDOW_FUNCTION: bool = false;
        }
    };
#[automatically_derived]
impl ::core::clone::Clone for ExcludeNoOthers {
    #[inline]
    fn clone(&self) -> ExcludeNoOthers { *self }
}
#[automatically_derived]
impl ::core::marker::Copy for ExcludeNoOthers { }
#[automatically_derived]
impl ::core::fmt::Debug for ExcludeNoOthers {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f, "ExcludeNoOthers")
    }
}
impl<DB> QueryFragment<DB> for ExcludeNoOthers where
    DB: crate::backend::Backend,
    Self: QueryFragment<DB, DB::WindowFrameExclusionSupport> {
    fn walk_ast<'b>(&'b self, pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        <Self as
                QueryFragment<DB,
                DB::WindowFrameExclusionSupport>>::walk_ast(self, pass)
    }
}
impl<DB>
    QueryFragment<DB,
    sql_dialect::window_frame_exclusion_support::FrameExclusionSupport> for
    ExcludeNoOthers where
    DB: crate::backend::Backend<WindowFrameExclusionSupport =
    sql_dialect::window_frame_exclusion_support::FrameExclusionSupport> {
    fn walk_ast<'b>(&'b self,
        mut pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        pass.push_sql("EXCLUDE NO OTHERS ");
        Ok(())
    }
}simple_frame_expr_with_bound!(
173    /// Exclude no rows from the frame windows
174    ///
175    /// This is the default behaviour if not specified
176    ExcludeNoOthers,
177    WindowFrameExclusionSupport,
178    sql_dialect::window_frame_exclusion_support::FrameExclusionSupport,
179    "EXCLUDE NO OTHERS "
180);
181
182/// A preceding frame clause expression with a fixed offset
183///
184/// Can be constructed via [`FrameBoundDsl::preceding`]
185#[derive(#[automatically_derived]
impl<T: ::core::clone::Clone> ::core::clone::Clone for OffsetPreceding<T> {
    #[inline]
    fn clone(&self) -> OffsetPreceding<T> {
        OffsetPreceding(::core::clone::Clone::clone(&self.0))
    }
}Clone, #[automatically_derived]
impl<T: ::core::marker::Copy> ::core::marker::Copy for OffsetPreceding<T> { }Copy, #[automatically_derived]
impl<T: ::core::fmt::Debug> ::core::fmt::Debug for OffsetPreceding<T> {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_tuple_field1_finish(f,
            "OffsetPreceding", &&self.0)
    }
}Debug)]
186pub struct OffsetPreceding<T = u64>(T);
187
188// manual impl as the derive makes this dependent on a `T: QueryId` impl
189// which is wrong
190impl QueryId for OffsetPreceding {
191    type QueryId = ();
192
193    const HAS_STATIC_QUERY_ID: bool = true;
194}
195
196impl<DB> QueryFragment<DB> for OffsetPreceding
197where
198    DB: crate::backend::Backend,
199{
200    fn walk_ast<'b>(
201        &'b self,
202        mut pass: crate::query_builder::AstPass<'_, 'b, DB>,
203    ) -> crate::QueryResult<()> {
204        // we cannot use binds here as mysql doesn't support it :(
205        // at least it's safe to just insert a number here
206        pass.push_sql(&self.0.to_string());
207        pass.push_sql(" PRECEDING ");
208        Ok(())
209    }
210}
211
212/// A following frame clause expression with a fixed offset
213///
214/// Can be constructed via [`FrameBoundDsl::following`]
215#[derive(#[automatically_derived]
impl<I: ::core::clone::Clone> ::core::clone::Clone for OffsetFollowing<I> {
    #[inline]
    fn clone(&self) -> OffsetFollowing<I> {
        OffsetFollowing(::core::clone::Clone::clone(&self.0))
    }
}Clone, #[automatically_derived]
impl<I: ::core::marker::Copy> ::core::marker::Copy for OffsetFollowing<I> { }Copy, #[automatically_derived]
impl<I: ::core::fmt::Debug> ::core::fmt::Debug for OffsetFollowing<I> {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_tuple_field1_finish(f,
            "OffsetFollowing", &&self.0)
    }
}Debug)]
216pub struct OffsetFollowing<I = u64>(I);
217
218// manual impl as the derive makes this dependent on a `T: QueryId` impl
219// which is wrong
220impl QueryId for OffsetFollowing {
221    type QueryId = ();
222
223    const HAS_STATIC_QUERY_ID: bool = true;
224}
225
226impl<DB> QueryFragment<DB> for OffsetFollowing
227where
228    DB: crate::backend::Backend,
229{
230    fn walk_ast<'b>(
231        &'b self,
232        mut pass: crate::query_builder::AstPass<'_, 'b, DB>,
233    ) -> crate::QueryResult<()> {
234        // we cannot use binds here as mysql doesn't support it :(
235        // at least it's safe to just insert a number here
236        pass.push_sql(&self.0.to_string());
237        pass.push_sql(" FOLLOWING ");
238        Ok(())
239    }
240}
241
242pub trait FrameDsl<F> {
243    type Output;
244
245    fn frame(self, expr: F) -> Self::Output;
246}
247
248#[diagnostic::on_unimplemented(
249    message = "`Groups` frame clauses require a ordered window function",
250    note = "call `.window_order(some_column)` first"
251)]
252pub trait ValidFrameClause<O> {}
253
254impl<O, Kind, Start, End, Exclusion> ValidFrameClause<O>
255    for BetweenFrame<Kind, Start, End, Exclusion>
256where
257    Kind: ValidFrameClause<O>,
258{
259}
260impl<O, Kind, Start, Exclusion> ValidFrameClause<O> for StartFrame<Kind, Start, Exclusion> where
261    Kind: ValidFrameClause<O>
262{
263}
264impl<O> ValidFrameClause<O> for Rows {}
265impl<O> ValidFrameClause<O> for Range {}
266impl<O> ValidFrameClause<Order<O, true>> for Groups {}
267
268impl<E, Fn, Filter, Frame, Partition, Order> FrameDsl<E>
269    for AggregateExpression<Fn, NoPrefix, NoOrder, Filter, OverClause<Partition, Order, Frame>>
270where
271    E: FrameClauseExpression,
272    E: ValidFrameClause<Order>,
273    Filter: ValidAggregateFilterForWindow<Fn, OverClause<Partition, Order, FrameClause<E>>>,
274{
275    type Output = AggregateExpression<
276        Fn,
277        NoPrefix,
278        NoOrder,
279        Filter,
280        OverClause<Partition, Order, FrameClause<E>>,
281    >;
282
283    fn frame(self, expr: E) -> Self::Output {
284        AggregateExpression {
285            prefix: self.prefix,
286            function: self.function,
287            order: self.order,
288            filter: self.filter,
289            window: OverClause {
290                partition_by: self.window.partition_by,
291                order: self.window.order,
292                frame_clause: FrameClause(expr),
293            },
294        }
295    }
296}
297
298impl<E, Fn, Filter> FrameDsl<E> for AggregateExpression<Fn, NoPrefix, NoOrder, Filter, NoWindow>
299where
300    E: FrameClauseExpression,
301    E: ValidFrameClause<NoOrder>,
302    Filter: ValidAggregateFilterForWindow<Fn, OverClause<NoPartition, NoOrder, FrameClause<E>>>,
303{
304    type Output = AggregateExpression<
305        Fn,
306        NoPrefix,
307        NoOrder,
308        Filter,
309        OverClause<NoPartition, NoOrder, FrameClause<E>>,
310    >;
311
312    fn frame(self, expr: E) -> Self::Output {
313        AggregateExpression {
314            prefix: self.prefix,
315            function: self.function,
316            order: self.order,
317            filter: self.filter,
318            window: OverClause {
319                partition_by: NoPartition,
320                order: NoOrder,
321                frame_clause: FrameClause(expr),
322            },
323        }
324    }
325}
326
327impl<E, Fn> FrameDsl<E> for Fn
328where
329    Fn: IsWindowFunction,
330    E: FrameClauseExpression,
331    E: ValidFrameClause<NoOrder>,
332{
333    type Output = AggregateExpression<
334        Fn,
335        NoPrefix,
336        NoOrder,
337        NoFilter,
338        OverClause<NoPartition, NoOrder, FrameClause<E>>,
339    >;
340
341    fn frame(self, expr: E) -> Self::Output {
342        AggregateExpression {
343            prefix: NoPrefix,
344            function: self,
345            order: NoOrder,
346            filter: NoFilter,
347            window: OverClause {
348                partition_by: NoPartition,
349                order: NoOrder,
350                frame_clause: FrameClause(expr),
351            },
352        }
353    }
354}
355
356pub trait FrameClauseExpression {}
357
358/// A marker trait for possible start frame expressions
359///
360/// See the list of types implementing this trait to understand
361/// what can be used in this position
362pub trait FrameClauseStartBound: Sealed {}
363
364/// A marker trait for possible end frame expressions
365///
366/// See the list of types implementing this trait to understand
367/// what can be used in this position
368pub trait FrameClauseEndBound: Sealed {}
369
370impl FrameClauseEndBound for UnboundedFollowing {}
371impl Sealed for UnboundedFollowing {}
372impl FrameClauseStartBound for UnboundedPreceding {}
373impl Sealed for UnboundedPreceding {}
374impl FrameClauseEndBound for CurrentRow {}
375impl FrameClauseStartBound for CurrentRow {}
376impl Sealed for CurrentRow {}
377impl FrameClauseEndBound for OffsetFollowing {}
378impl Sealed for OffsetFollowing {}
379impl FrameClauseStartBound for OffsetPreceding {}
380impl Sealed for OffsetPreceding {}
381
382/// A marker trait for possible frame exclusion expressions
383///
384/// See the list of types implementing this trait to understand
385/// what can be used in this position
386pub trait FrameClauseExclusion: Sealed {}
387
388impl FrameClauseExclusion for ExcludeGroup {}
389impl Sealed for ExcludeGroup {}
390impl FrameClauseExclusion for ExcludeNoOthers {}
391impl Sealed for ExcludeNoOthers {}
392impl FrameClauseExclusion for ExcludeTies {}
393impl Sealed for ExcludeTies {}
394impl FrameClauseExclusion for ExcludeCurrentRow {}
395impl Sealed for ExcludeCurrentRow {}
396
397/// Construct a frame clause for window functions from an integer
398pub trait FrameBoundDsl {
399    /// Use the preceding frame clause specification
400    fn preceding(self) -> OffsetPreceding;
401
402    /// Use the following frame clause specification
403    fn following(self) -> OffsetFollowing;
404}
405
406impl FrameBoundDsl for u64 {
407    fn preceding(self) -> OffsetPreceding {
408        OffsetPreceding(self)
409    }
410
411    fn following(self) -> OffsetFollowing {
412        OffsetFollowing(self)
413    }
414}
415// TODO: We might want to implement
416// it for datetime and date intervals?
417// The postgres documentation indicates that
418// something like `RANGE BETWEEN '1 day' PRECEDING AND '10 days' FOLLOWING`
419// is valid
420
421pub struct NoExclusion;
#[automatically_derived]
impl ::core::fmt::Debug for NoExclusion {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f, "NoExclusion")
    }
}
#[automatically_derived]
impl ::core::clone::Clone for NoExclusion {
    #[inline]
    fn clone(&self) -> NoExclusion { *self }
}
#[automatically_derived]
impl ::core::marker::Copy for NoExclusion { }
const _: () =
    {
        use diesel;
        #[allow(non_camel_case_types)]
        impl diesel::query_builder::QueryId for NoExclusion {
            type QueryId = NoExclusion<>;
            const HAS_STATIC_QUERY_ID: bool = true;
            const IS_WINDOW_FUNCTION: bool = false;
        }
    };
impl<DB> crate::query_builder::QueryFragment<DB> for NoExclusion where
    DB: crate::backend::Backend + crate::backend::DieselReserveSpecialization
    {
    fn walk_ast<'b>(&'b self,
        _pass: crate::query_builder::AstPass<'_, 'b, DB>)
        -> crate::QueryResult<()> {
        Ok(())
    }
}empty_clause!(NoExclusion);
422
423#[derive(const _: () =
    {
        use diesel;
        #[allow(non_camel_case_types)]
        impl<Kind: diesel::query_builder::QueryId,
            Start: diesel::query_builder::QueryId,
            Exclusion: diesel::query_builder::QueryId>
            diesel::query_builder::QueryId for
            StartFrame<Kind, Start, Exclusion> {
            type QueryId =
                StartFrame<<Kind as diesel::query_builder::QueryId>::QueryId,
                <Start as diesel::query_builder::QueryId>::QueryId,
                <Exclusion as diesel::query_builder::QueryId>::QueryId>;
            const HAS_STATIC_QUERY_ID: bool =
                <Kind as diesel::query_builder::QueryId>::HAS_STATIC_QUERY_ID
                            &&
                            <Start as
                                diesel::query_builder::QueryId>::HAS_STATIC_QUERY_ID &&
                        <Exclusion as
                            diesel::query_builder::QueryId>::HAS_STATIC_QUERY_ID &&
                    true;
            const IS_WINDOW_FUNCTION: bool =
                <Kind as diesel::query_builder::QueryId>::IS_WINDOW_FUNCTION
                            ||
                            <Start as
                                diesel::query_builder::QueryId>::IS_WINDOW_FUNCTION ||
                        <Exclusion as
                            diesel::query_builder::QueryId>::IS_WINDOW_FUNCTION ||
                    false;
        }
    };QueryId, #[automatically_derived]
impl<Kind: ::core::marker::Copy, Start: ::core::marker::Copy,
    Exclusion: ::core::marker::Copy> ::core::marker::Copy for
    StartFrame<Kind, Start, Exclusion> {
}Copy, #[automatically_derived]
impl<Kind: ::core::clone::Clone, Start: ::core::clone::Clone,
    Exclusion: ::core::clone::Clone> ::core::clone::Clone for
    StartFrame<Kind, Start, Exclusion> {
    #[inline]
    fn clone(&self) -> StartFrame<Kind, Start, Exclusion> {
        StartFrame {
            kind: ::core::clone::Clone::clone(&self.kind),
            start: ::core::clone::Clone::clone(&self.start),
            exclusion: ::core::clone::Clone::clone(&self.exclusion),
        }
    }
}Clone, #[automatically_derived]
impl<Kind: ::core::fmt::Debug, Start: ::core::fmt::Debug,
    Exclusion: ::core::fmt::Debug> ::core::fmt::Debug for
    StartFrame<Kind, Start, Exclusion> {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field3_finish(f, "StartFrame",
            "kind", &self.kind, "start", &self.start, "exclusion",
            &&self.exclusion)
    }
}Debug)]
424pub struct StartFrame<Kind, Start, Exclusion = NoExclusion> {
425    kind: Kind,
426    start: Start,
427    exclusion: Exclusion,
428}
429
430impl<Kind, Start, Exclusion, DB> QueryFragment<DB> for StartFrame<Kind, Start, Exclusion>
431where
432    Kind: QueryFragment<DB>,
433    Start: QueryFragment<DB>,
434    Exclusion: QueryFragment<DB>,
435    DB: crate::backend::Backend,
436{
437    fn walk_ast<'b>(
438        &'b self,
439        mut pass: crate::query_builder::AstPass<'_, 'b, DB>,
440    ) -> crate::QueryResult<()> {
441        self.kind.walk_ast(pass.reborrow())?;
442        self.start.walk_ast(pass.reborrow())?;
443        self.exclusion.walk_ast(pass.reborrow())?;
444        Ok(())
445    }
446}
447
448impl<Kind, Start, Exclusion> FrameClauseExpression for StartFrame<Kind, Start, Exclusion> {}
449
450#[derive(const _: () =
    {
        use diesel;
        #[allow(non_camel_case_types)]
        impl<Kind: diesel::query_builder::QueryId,
            Start: diesel::query_builder::QueryId,
            End: diesel::query_builder::QueryId,
            Exclusion: diesel::query_builder::QueryId>
            diesel::query_builder::QueryId for
            BetweenFrame<Kind, Start, End, Exclusion> {
            type QueryId =
                BetweenFrame<<Kind as
                diesel::query_builder::QueryId>::QueryId,
                <Start as diesel::query_builder::QueryId>::QueryId,
                <End as diesel::query_builder::QueryId>::QueryId,
                <Exclusion as diesel::query_builder::QueryId>::QueryId>;
            const HAS_STATIC_QUERY_ID: bool =
                <Kind as diesel::query_builder::QueryId>::HAS_STATIC_QUERY_ID
                                &&
                                <Start as
                                    diesel::query_builder::QueryId>::HAS_STATIC_QUERY_ID &&
                            <End as diesel::query_builder::QueryId>::HAS_STATIC_QUERY_ID
                        &&
                        <Exclusion as
                            diesel::query_builder::QueryId>::HAS_STATIC_QUERY_ID &&
                    true;
            const IS_WINDOW_FUNCTION: bool =
                <Kind as diesel::query_builder::QueryId>::IS_WINDOW_FUNCTION
                                ||
                                <Start as
                                    diesel::query_builder::QueryId>::IS_WINDOW_FUNCTION ||
                            <End as diesel::query_builder::QueryId>::IS_WINDOW_FUNCTION
                        ||
                        <Exclusion as
                            diesel::query_builder::QueryId>::IS_WINDOW_FUNCTION ||
                    false;
        }
    };QueryId, #[automatically_derived]
impl<Kind: ::core::marker::Copy, Start: ::core::marker::Copy,
    End: ::core::marker::Copy, Exclusion: ::core::marker::Copy>
    ::core::marker::Copy for BetweenFrame<Kind, Start, End, Exclusion> {
}Copy, #[automatically_derived]
impl<Kind: ::core::clone::Clone, Start: ::core::clone::Clone,
    End: ::core::clone::Clone, Exclusion: ::core::clone::Clone>
    ::core::clone::Clone for BetweenFrame<Kind, Start, End, Exclusion> {
    #[inline]
    fn clone(&self) -> BetweenFrame<Kind, Start, End, Exclusion> {
        BetweenFrame {
            kind: ::core::clone::Clone::clone(&self.kind),
            start: ::core::clone::Clone::clone(&self.start),
            end: ::core::clone::Clone::clone(&self.end),
            exclusion: ::core::clone::Clone::clone(&self.exclusion),
        }
    }
}Clone, #[automatically_derived]
impl<Kind: ::core::fmt::Debug, Start: ::core::fmt::Debug,
    End: ::core::fmt::Debug, Exclusion: ::core::fmt::Debug> ::core::fmt::Debug
    for BetweenFrame<Kind, Start, End, Exclusion> {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field4_finish(f, "BetweenFrame",
            "kind", &self.kind, "start", &self.start, "end", &self.end,
            "exclusion", &&self.exclusion)
    }
}Debug)]
451pub struct BetweenFrame<Kind, Start, End, Exclusion = NoExclusion> {
452    kind: Kind,
453    start: Start,
454    end: End,
455    exclusion: Exclusion,
456}
457
458impl<Kind, Start, End, Exclusion, DB> QueryFragment<DB>
459    for BetweenFrame<Kind, Start, End, Exclusion>
460where
461    Kind: QueryFragment<DB>,
462    Start: QueryFragment<DB>,
463    End: QueryFragment<DB>,
464    Exclusion: QueryFragment<DB>,
465    DB: crate::backend::Backend,
466{
467    fn walk_ast<'b>(
468        &'b self,
469        mut pass: crate::query_builder::AstPass<'_, 'b, DB>,
470    ) -> crate::QueryResult<()> {
471        self.kind.walk_ast(pass.reborrow())?;
472        pass.push_sql(" BETWEEN ");
473        self.start.walk_ast(pass.reborrow())?;
474        pass.push_sql(" AND ");
475        self.end.walk_ast(pass.reborrow())?;
476        self.exclusion.walk_ast(pass.reborrow())?;
477        Ok(())
478    }
479}
480
481impl<Kind, Start, End, Exclusion> FrameClauseExpression
482    for BetweenFrame<Kind, Start, End, Exclusion>
483{
484}
485
486pub trait FrameClauseDslHelper: Sized {}
487
488/// Construct a frame clause for window functions
489pub trait FrameClauseDsl: FrameClauseDslHelper {
490    /// Construct a frame clause with a starting bound
491    fn frame_start_with<E>(self, start: E) -> super::dsl::FrameStartWith<Self, E>
492    where
493        E: FrameClauseStartBound,
494    {
495        StartFrame {
496            kind: self,
497            start,
498            exclusion: NoExclusion,
499        }
500    }
501
502    /// Construct a frame clause with a starting bound and an exclusion condition
503    fn frame_start_with_exclusion<E1, E2>(
504        self,
505        start: E1,
506        exclusion: E2,
507    ) -> super::dsl::FrameStartWithExclusion<Self, E1, E2>
508    where
509        E1: FrameClauseStartBound,
510        E2: FrameClauseExclusion,
511    {
512        StartFrame {
513            kind: self,
514            start,
515            exclusion,
516        }
517    }
518
519    /// Construct a between frame clause with a starting and end bound
520    fn frame_between<E1, E2>(self, start: E1, end: E2) -> super::dsl::FrameBetween<Self, E1, E2>
521    where
522        E1: FrameClauseStartBound,
523        E2: FrameClauseEndBound,
524    {
525        BetweenFrame {
526            kind: self,
527            start,
528            end,
529            exclusion: NoExclusion,
530        }
531    }
532
533    /// Construct a between frame clause with a starting and end bound  with an exclusion condition
534    fn frame_between_with_exclusion<E1, E2, E3>(
535        self,
536        start: E1,
537        end: E2,
538        exclusion: E3,
539    ) -> super::dsl::FrameBetweenWithExclusion<Self, E1, E2, E3>
540    where
541        E1: FrameClauseStartBound,
542        E2: FrameClauseEndBound,
543        E3: FrameClauseExclusion,
544    {
545        BetweenFrame {
546            kind: self,
547            start,
548            end,
549            exclusion,
550        }
551    }
552}
553
554impl<T> FrameClauseDsl for T where T: FrameClauseDslHelper {}
555
556impl FrameClauseDslHelper for Range {}
557impl FrameClauseDslHelper for Rows {}
558impl FrameClauseDslHelper for Groups {}
559
560pub trait Sealed {}