diesel/query_source/aliasing/
macros.rs

1/// Declare a new alias for a table
2///
3/// This macro creates a value of the type [`Alias`](super::Alias)
4///
5/// Example usage
6/// -------------
7/// ```rust
8/// # include!("../../doctest_setup.rs");
9/// fn main() {
10///     use schema::users;
11///     let connection = &mut establish_connection();
12///     let (users1, users2) = diesel::alias!(schema::users as user1, schema::users as user2);
13///     let res = users1
14///         .inner_join(users2.on(users1.field(users::id).eq(users2.field(users::id))))
15///         .select((users1.fields((users::id, users::name)), users2.field(users::name)))
16///         .order_by(users2.field(users::id))
17///         .load::<((i32, String), String)>(connection);
18///     assert_eq!(
19///         res,
20///         Ok(vec![
21///             ((1, "Sean".to_owned()), "Sean".to_owned()),
22///             ((2, "Tess".to_owned()), "Tess".to_owned()),
23///         ]),
24///     );
25/// }
26/// ```
27///
28///
29/// Make type expressible
30/// ---------------------
31/// It may sometimes be useful to declare an alias at the module level, in such a way that the type
32/// of a query using it can be expressed (to not declare it anonymously).
33///
34/// This can be achieved in the following way
35/// ```rust
36/// # include!("../../doctest_setup.rs");
37/// use diesel::{query_source::Alias, dsl};
38///
39/// diesel::alias!(schema::users as users_alias: UsersAlias);
40/// // or
41/// diesel::alias!{
42///     pub const USERS_ALIAS_2: Alias<UsersAlias2> = schema::users as users_alias_2;
43/// }
44///
45/// fn some_function_that_returns_a_query_fragment(
46/// ) -> dsl::InnerJoin<schema::posts::table, Alias<UsersAlias>>
47/// {
48///     schema::posts::table.inner_join(users_alias)
49/// }
50/// # fn main() {
51/// #     some_function_that_returns_a_query_fragment();
52/// #     schema::posts::table.inner_join(USERS_ALIAS_2);
53/// # }
54/// ```
55///
56/// Note that you may also use this form within a function, in the following way:
57/// ```rust
58/// # include!("../../doctest_setup.rs");
59/// fn main() {
60///     diesel::alias!(schema::users as users_alias: UsersAlias);
61///     users_alias.inner_join(schema::posts::table);
62/// }
63/// ```
64///
65/// Troubleshooting and limitations
66/// -------------------------------
67/// If you encounter a **compilation error** where "the trait
68/// `AppearsInFromClause<Alias<your_alias>>` is not implemented", when trying to use two aliases to
69/// the same table within a single query note the following two limitations:
70///  - You will need to declare these in a single `alias!` call.
71///  - The path to the table module will have to be expressed in the exact same
72///    manner. (That is, you can do `alias!(schema::users as user1, schema::users as user2)`
73///    or `alias!(users as user1, users as user2)`, but not
74///    `alias!(schema::users as user1, users as user2)`)
75#[macro_export]
76macro_rules! alias {
77    ($($($table: ident)::+ as $alias: ident),* $(,)?) => {{
78        $crate::alias!(NoConst $($($table)::+ as $alias: $alias,)*);
79        ($($crate::query_source::Alias::<$alias>::default()),*)
80    }};
81    ($($($table: ident)::+ as $alias_name: ident: $alias_ty: ident),* $(,)?) => {
82        $crate::alias! {
83            $(
84                pub const $alias_name: Alias<$alias_ty> = $($table)::+ as $alias_name;
85            )*
86        }
87    };
88    ($($vis: vis const $const_name: ident: Alias<$alias_ty: ident> = $($table: ident)::+ as $alias_sql_name: ident);* $(;)?) => {
89        $crate::alias!(NoConst $($($table)::+ as $alias_sql_name: $vis $alias_ty,)*);
90        $(
91            #[allow(non_upper_case_globals)]
92            $vis const $const_name: $crate::query_source::Alias::<$alias_ty> =
93                $crate::query_source::Alias::new($alias_ty { table: $($table)::+::table });
94
95            #[allow(non_camel_case_types)]
96            $vis type $const_name = $crate::query_source::Alias::<$alias_ty>;
97        )*
98    };
99    (NoConst $($($table: ident)::+ as $alias_sql_name: ident: $vis: vis $alias_ty: ident),* $(,)?) => {
100        $(
101            #[allow(non_camel_case_types)]
102            #[derive(Debug, Clone, Copy)]
103            $vis struct $alias_ty {
104                table: $($table)::+::table,
105            }
106
107            impl $crate::query_source::AliasSource for $alias_ty {
108                const NAME: &'static str = stringify!($alias_sql_name);
109                type Target = $($table)::+::table;
110                fn target(&self) -> &Self::Target { &self.table }
111            }
112
113            impl ::std::default::Default for $alias_ty {
114                fn default() -> Self {
115                    Self { table: $($table)::+::table }
116                }
117            }
118
119            // impl AppearsInFromClause<Alias<$alias>> for Alias<$alias>
120            impl $crate::internal::alias_macro::AliasAliasAppearsInFromClauseSameTable<$alias_ty, $($table)::+::table> for $alias_ty {
121                type Count = $crate::query_source::Once;
122            }
123        )*
124        $crate::__internal_alias_helper!($(table_ty = $($table)::+::table, table_tt = ($($table)::+), alias_ty = $alias_ty, alias_sql_name = $alias_sql_name;)*);
125    };
126}
127
128#[macro_export]
129#[doc(hidden)]
130/// This only exists to hide internals from the doc
131// The `($left_sql_name) != ($right_sql_name)` condition is not perfect because a user could still
132// cause runtime errors by declaring two aliases to different tables with the same name then use
133// them in the same query, but that would almost have to be voluntary, so this alone should prevent
134// most mistakes.
135macro_rules! __internal_alias_helper {
136    (
137        table_ty = $left_table_ty: ty, table_tt = $left_table_tt: tt, alias_ty = $left_alias: ident, alias_sql_name = $left_sql_name: ident;
138        $(table_ty = $right_table_ty: ty, table_tt = $right_table_tt: tt, alias_ty = $right_alias: ident, alias_sql_name = $right_sql_name: ident;)+
139    ) => {
140        $(
141            $crate::static_cond!{if ($left_table_tt) == ($right_table_tt) {
142                $crate::static_cond!{if ($left_sql_name) != ($right_sql_name) {
143                    impl $crate::internal::alias_macro::AliasAliasAppearsInFromClauseSameTable<$left_alias, $left_table_ty>
144                        for $right_alias
145                    {
146                        type Count = $crate::query_source::Never;
147                    }
148                    impl $crate::internal::alias_macro::AliasAliasAppearsInFromClauseSameTable<$right_alias, $left_table_ty>
149                        for $left_alias
150                    {
151                        type Count = $crate::query_source::Never;
152                    }
153                }}
154            }}
155        )*
156        $crate::__internal_alias_helper!($(table_ty = $right_table_ty, table_tt = $right_table_tt, alias_ty = $right_alias, alias_sql_name = $right_sql_name;)+);
157    };
158
159    (table_ty = $left_table_ty: ty, table_tt = $left_table_tt: tt, alias_ty = $left_alias: ident, alias_sql_name = $left_sql_name: ident;) => {};
160    () => {};
161}