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}