diesel/query_dsl/
positional_order_dsl.rs
1use crate::backend::{Backend, DieselReserveSpecialization};
2use crate::expression::helper_types::{Asc, Desc};
3use crate::query_builder::combination_clause::CombinationClause;
4use crate::query_builder::{AstPass, Query, QueryFragment, QueryId};
5use crate::{QueryResult, RunQueryDsl};
6
7pub trait PositionalOrderDsl<Expr: Order>: Sized {
15 fn positional_order_by(self, expr: Expr) -> PositionalOrderClause<Self, Expr::Fragment> {
16 PositionalOrderClause {
17 source: self,
18 expr: expr.into_fragment(),
19 }
20 }
21}
22
23#[derive(Debug, Clone, Copy, QueryId)]
24pub struct PositionalOrderClause<Source, Expr> {
25 source: Source,
26 expr: Expr,
27}
28
29impl<Combinator, Rule, Source, Rhs, Expr: Order> PositionalOrderDsl<Expr>
30 for CombinationClause<Combinator, Rule, Source, Rhs>
31{
32}
33
34impl<Source, Expr> Query for PositionalOrderClause<Source, Expr>
35where
36 Source: Query,
37{
38 type SqlType = Source::SqlType;
39}
40
41impl<Source, Expr, Conn> RunQueryDsl<Conn> for PositionalOrderClause<Source, Expr> {}
42
43impl<Source, Expr, DB> QueryFragment<DB> for PositionalOrderClause<Source, Expr>
44where
45 DB: Backend + DieselReserveSpecialization,
46 Source: QueryFragment<DB>,
47 Expr: QueryFragment<DB>,
48{
49 fn walk_ast<'b>(&'b self, mut pass: AstPass<'_, 'b, DB>) -> QueryResult<()> {
50 self.source.walk_ast(pass.reborrow())?;
51 pass.push_sql(" ORDER BY ");
52 self.expr.walk_ast(pass)
53 }
54}
55
56#[derive(Debug, Clone, Copy, QueryId)]
57pub struct OrderColumn(u32);
58
59impl<DB: Backend> QueryFragment<DB> for OrderColumn {
60 fn walk_ast<'b>(&'b self, mut pass: AstPass<'_, 'b, DB>) -> QueryResult<()> {
61 pass.push_sql(&self.0.to_string());
62 Ok(())
63 }
64}
65
66impl From<u32> for OrderColumn {
67 fn from(order: u32) -> Self {
68 OrderColumn(order)
69 }
70}
71
72pub trait IntoOrderColumn: Into<OrderColumn> {
73 fn asc(self) -> Asc<OrderColumn> {
74 Asc { expr: self.into() }
75 }
76 fn desc(self) -> Desc<OrderColumn> {
77 Desc { expr: self.into() }
78 }
79}
80
81impl<T> IntoOrderColumn for T where T: Into<OrderColumn> {}
82
83pub trait Order: Copy {
84 type Fragment;
85
86 fn into_fragment(self) -> Self::Fragment;
87}
88
89impl<T: Into<OrderColumn> + Copy> Order for T {
90 type Fragment = OrderColumn;
91
92 fn into_fragment(self) -> Self::Fragment {
93 self.into()
94 }
95}
96
97impl Order for Asc<OrderColumn> {
98 type Fragment = Asc<OrderColumn>;
99
100 fn into_fragment(self) -> Self::Fragment {
101 self
102 }
103}
104
105impl Order for Desc<OrderColumn> {
106 type Fragment = Desc<OrderColumn>;
107
108 fn into_fragment(self) -> Self::Fragment {
109 self
110 }
111}
112
113macro_rules! impl_order_for_all_tuples {
114 ($(
115 $unused1:tt {
116 $(($idx:tt) -> $T:ident, $unused2:ident, $unused3:tt,)+
117 }
118 )+) => {
119 $(
120 impl<$($T: Order),+> Order for ($($T,)+) {
121 type Fragment = ($(<$T as Order>::Fragment,)+);
122
123 fn into_fragment(self) -> Self::Fragment {
124 ($(self.$idx.into_fragment(),)+)
125 }
126 }
127 )+
128 };
129}
130
131diesel_derives::__diesel_for_each_tuple!(impl_order_for_all_tuples);