From 03f7909391a24f23d515a17e00b0c38b56c6bdf4 Mon Sep 17 00:00:00 2001 From: Georg Semmler Date: Fri, 20 Nov 2020 14:50:43 +0100 Subject: [PATCH 1/4] Make `.group_by` part of our public API I think we figured out all implementation details, so we can make `GroupByDsl` part of our public API. This commit adds the necessary documentation + a helper type as it exists for all other `QueryDsl` methods. Fix #210 --- diesel/src/lib.rs | 5 +-- diesel/src/query_dsl/group_by_dsl.rs | 14 +++----- diesel/src/query_dsl/mod.rs | 53 ++++++++++++++++++++++++++-- 3 files changed, 58 insertions(+), 14 deletions(-) diff --git a/diesel/src/lib.rs b/diesel/src/lib.rs index 44b770a60375..4f3374a3fab6 100644 --- a/diesel/src/lib.rs +++ b/diesel/src/lib.rs @@ -303,6 +303,9 @@ pub mod helper_types { /// Represents the return type of `.nullable()` pub type NullableSelect = ::Output; + + /// Represents the return type of `.group_by(expr)` + pub type GroupBy = >::Output; } pub mod prelude { @@ -332,8 +335,6 @@ pub mod prelude { pub use crate::query_builder::AsChangeset; #[doc(inline)] pub use crate::query_builder::DecoratableTarget; - #[doc(hidden)] - pub use crate::query_dsl::GroupByDsl; #[doc(inline)] pub use crate::query_dsl::{BelongingToDsl, JoinOnDsl, QueryDsl, RunQueryDsl, SaveChangesDsl}; #[doc(inline)] diff --git a/diesel/src/query_dsl/group_by_dsl.rs b/diesel/src/query_dsl/group_by_dsl.rs index a79c145d606f..7c798173dd13 100644 --- a/diesel/src/query_dsl/group_by_dsl.rs +++ b/diesel/src/query_dsl/group_by_dsl.rs @@ -2,17 +2,13 @@ use crate::expression::Expression; use crate::query_builder::AsQuery; use crate::query_source::Table; -/// This trait is not yet part of Diesel's public API. It may change in the -/// future without a major version bump. +/// The `group_by` method /// -/// This trait exists as a stop-gap for users who need to use `GROUP BY` in -/// their queries, so that they are not forced to drop entirely to raw SQL. The -/// arguments to `group_by` are not checked, nor is the select statement -/// forced to be valid. +/// This trait should not be relied on directly by most apps. Its behavior is +/// provided by [`QueryDsl`]. However, you may need a where clause on this trait +/// to call `group_by` from generic code. /// -/// Since Diesel otherwise assumes that you have no `GROUP BY` clause (which -/// would mean that mixing an aggregate and non aggregate expression in the same -/// query is an error), you may need to use `sql` for your select clause. +/// [`QueryDsl`]: ../trait.QueryDsl.html pub trait GroupByDsl { /// The type returned by `.group_by` type Output; diff --git a/diesel/src/query_dsl/mod.rs b/diesel/src/query_dsl/mod.rs index cc79c53617cc..d61b94fcbdff 100644 --- a/diesel/src/query_dsl/mod.rs +++ b/diesel/src/query_dsl/mod.rs @@ -44,8 +44,6 @@ pub mod select_dsl; mod single_value_dsl; pub use self::belonging_to_dsl::BelongingToDsl; -#[doc(hidden)] -pub use self::group_by_dsl::GroupByDsl; pub use self::join_dsl::{InternalJoinDsl, JoinOnDsl, JoinWithImplicitOnClause}; #[doc(hidden)] pub use self::load_dsl::LoadQuery; @@ -62,6 +60,7 @@ pub mod methods { pub use super::distinct_dsl::*; #[doc(inline)] pub use super::filter_dsl::*; + pub use super::group_by_dsl::GroupByDsl; pub use super::limit_dsl::LimitDsl; pub use super::load_dsl::{ExecuteDsl, LoadQuery}; pub use super::locking_dsl::{LockingDsl, ModifyLockDsl}; @@ -549,7 +548,7 @@ pub trait QueryDsl: Sized { /// Sets the order clause of a query. /// - /// If there was already a order clause, it will be overridden. See + /// If there was already an order clause, it will be overridden. See /// also: /// [`.desc()`](../expression_methods/trait.ExpressionMethods.html#method.desc) /// and @@ -770,6 +769,54 @@ pub trait QueryDsl: Sized { methods::OffsetDsl::offset(self, offset) } + /// Sets the `group by` clause of a query. + /// + /// **Note:** Queries having a `group by` clause require a custom select clause. + /// Use `QueryDsl::select()` to specify one + /// + /// If there was already a group by clause, it will be overridden. + /// Ordering by multiple columns can be achieved by passing a tuple of those + /// columns. + /// + /// Diesel follows postgresql's group by semantic, this means any column + /// appearing in a group by clause is considered to be aggregated. If a + /// primary key is part of the group by clause every column from the + /// corresponding table is considerd to be aggregated. Select clauses + /// cannot mix aggregated and non aggregated expressions. + /// + /// For group by clauses containing columns from more than one table it + /// is required to call [`allow_columns_to_appear_in_same_group_by_clause!`] + /// + /// [`allow_columns_to_appear_in_same_group_by_clause!`]: ../macro.allow_columns_to_appear_in_same_group_by_clause.html + /// + /// # Examples + /// ```rust + /// # include!("../doctest_setup.rs"); + /// # fn main() { + /// # run_test(); + /// # } + /// # + /// # fn run_test() -> QueryResult<()> { + /// # use crate::schema::{users, posts}; + /// # use diesel::dsl::count; + /// # let connection = establish_connection(); + /// let data = users::table.inner_join(posts::table) + /// .group_by(users::id) + /// .select((users::name, count(posts::id))) + /// .load::<(String, i64)>(&connection)?; + /// + /// assert_eq!(vec![(String::from("Sean"), 2), (String::from("Tess"), 1)], data); + /// # Ok(()) + /// # } + /// ``` + fn group_by(self, group_by: GB) -> GroupBy + where + GB: Expression, + Self: methods::GroupByDsl, + { + methods::GroupByDsl::group_by(self, group_by) + } + /// Adds `FOR UPDATE` to the end of the select statement. /// /// This method is only available for MySQL and PostgreSQL. SQLite does not From a2e21be573fdab0510a350724afbd2da05e7a305 Mon Sep 17 00:00:00 2001 From: Georg Semmler Date: Fri, 20 Nov 2020 14:54:25 +0100 Subject: [PATCH 2/4] Use typedefs from `diesel::dsl` for join methods on `QueryDsl` --- diesel/src/query_dsl/mod.rs | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/diesel/src/query_dsl/mod.rs b/diesel/src/query_dsl/mod.rs index d61b94fcbdff..5433fdc7cf60 100644 --- a/diesel/src/query_dsl/mod.rs +++ b/diesel/src/query_dsl/mod.rs @@ -416,7 +416,7 @@ pub trait QueryDsl: Sized { /// assert_eq!(Ok(expected_data), data); /// # } /// ``` - fn inner_join(self, rhs: Rhs) -> Self::Output + fn inner_join(self, rhs: Rhs) -> InnerJoin where Self: JoinWithImplicitOnClause, { @@ -429,7 +429,7 @@ pub trait QueryDsl: Sized { /// instead. See [`inner_join`] for usage examples. /// /// [`inner_join`]: #method.inner_join - fn left_outer_join(self, rhs: Rhs) -> Self::Output + fn left_outer_join(self, rhs: Rhs) -> LeftJoin where Self: JoinWithImplicitOnClause, { @@ -439,7 +439,7 @@ pub trait QueryDsl: Sized { /// Alias for [`left_outer_join`]. /// /// [`left_outer_join`]: #method.left_outer_join - fn left_join(self, rhs: Rhs) -> Self::Output + fn left_join(self, rhs: Rhs) -> LeftJoin where Self: JoinWithImplicitOnClause, { From 4ef8c77248c80458a31d3ca6c7daf0c0fbd34579 Mon Sep 17 00:00:00 2001 From: Georg Semmler Date: Thu, 26 Nov 2020 19:29:14 +0100 Subject: [PATCH 3/4] Improve our compile error messages for some invalid queries Instead of producing an overflowing trait bound error show a concrete error message what's wrong for certain query combination. This requires to workaround /~https://github.com/rust-lang/rust/issues/34260 in our single method query dsl implementations: * `GroupByDsl` * `BoxedDsl` * `DistinctDsl` * `DistinctOnDsl` * `LockingDsl` --- diesel/src/query_dsl/boxed_dsl.rs | 15 +++++-- diesel/src/query_dsl/distinct_dsl.rs | 28 ++++++++----- diesel/src/query_dsl/group_by_dsl.rs | 16 +++++--- diesel/src/query_dsl/locking_dsl.rs | 11 +++-- diesel/src/query_dsl/mod.rs | 1 + .../boxed_queries_and_group_by.rs | 18 ++++---- ...pdate_cannot_be_mixed_with_some_clauses.rs | 41 ++++++++++++------- 7 files changed, 85 insertions(+), 45 deletions(-) diff --git a/diesel/src/query_dsl/boxed_dsl.rs b/diesel/src/query_dsl/boxed_dsl.rs index 10d18e9ee36d..701bcce37a7f 100644 --- a/diesel/src/query_dsl/boxed_dsl.rs +++ b/diesel/src/query_dsl/boxed_dsl.rs @@ -1,5 +1,10 @@ +use crate::dsl; +use crate::expression::TypedExpressionType; +use crate::expression::ValidGrouping; use crate::query_builder::AsQuery; +use crate::query_builder::SelectStatement; use crate::query_source::Table; +use crate::Expression; /// The `into_boxed` method /// @@ -13,15 +18,17 @@ pub trait BoxedDsl<'a, DB> { type Output; /// See the trait documentation. - fn internal_into_boxed(self) -> Self::Output; + fn internal_into_boxed(self) -> dsl::IntoBoxed<'a, Self, DB>; } impl<'a, T, DB> BoxedDsl<'a, DB> for T where - T: Table + AsQuery, - T::Query: BoxedDsl<'a, DB>, + T: Table + AsQuery>, + SelectStatement: BoxedDsl<'a, DB>, + T::DefaultSelection: Expression + ValidGrouping<()>, + T::SqlType: TypedExpressionType, { - type Output = >::Output; + type Output = dsl::IntoBoxed<'a, SelectStatement, DB>; fn internal_into_boxed(self) -> Self::Output { self.as_query().internal_into_boxed() diff --git a/diesel/src/query_dsl/distinct_dsl.rs b/diesel/src/query_dsl/distinct_dsl.rs index cf82aad18c0c..85dbd2e4c294 100644 --- a/diesel/src/query_dsl/distinct_dsl.rs +++ b/diesel/src/query_dsl/distinct_dsl.rs @@ -1,6 +1,12 @@ +use crate::dsl; #[cfg(feature = "postgres")] use crate::expression::SelectableExpression; +use crate::expression::TypedExpressionType; +use crate::expression::ValidGrouping; +use crate::query_builder::AsQuery; +use crate::query_builder::SelectStatement; use crate::query_source::Table; +use crate::Expression; /// The `distinct` method /// @@ -14,17 +20,18 @@ pub trait DistinctDsl { type Output; /// See the trait documentation. - fn distinct(self) -> Self::Output; + fn distinct(self) -> dsl::Distinct; } impl DistinctDsl for T where - T: Table, - T::Query: DistinctDsl, + T: Table + AsQuery>, + T::DefaultSelection: Expression + ValidGrouping<()>, + T::SqlType: TypedExpressionType, { - type Output = ::Output; + type Output = dsl::Distinct>; - fn distinct(self) -> Self::Output { + fn distinct(self) -> dsl::Distinct> { self.as_query().distinct() } } @@ -42,19 +49,20 @@ pub trait DistinctOnDsl { type Output; /// See the trait documentation - fn distinct_on(self, selection: Selection) -> Self::Output; + fn distinct_on(self, selection: Selection) -> dsl::DistinctOn; } #[cfg(feature = "postgres")] impl DistinctOnDsl for T where Selection: SelectableExpression, - T: Table, - T::Query: DistinctOnDsl, + T: Table + AsQuery>, + T::DefaultSelection: Expression + ValidGrouping<()>, + T::SqlType: TypedExpressionType, { - type Output = >::Output; + type Output = dsl::DistinctOn, Selection>; - fn distinct_on(self, selection: Selection) -> Self::Output { + fn distinct_on(self, selection: Selection) -> dsl::DistinctOn { self.as_query().distinct_on(selection) } } diff --git a/diesel/src/query_dsl/group_by_dsl.rs b/diesel/src/query_dsl/group_by_dsl.rs index 7c798173dd13..517ec2b32c1d 100644 --- a/diesel/src/query_dsl/group_by_dsl.rs +++ b/diesel/src/query_dsl/group_by_dsl.rs @@ -1,5 +1,8 @@ +use crate::dsl; use crate::expression::Expression; -use crate::query_builder::AsQuery; +use crate::expression::TypedExpressionType; +use crate::expression::ValidGrouping; +use crate::query_builder::{AsQuery, SelectStatement}; use crate::query_source::Table; /// The `group_by` method @@ -14,18 +17,19 @@ pub trait GroupByDsl { type Output; /// See the trait documentation. - fn group_by(self, expr: Expr) -> Self::Output; + fn group_by(self, expr: Expr) -> dsl::GroupBy; } impl GroupByDsl for T where Expr: Expression, - T: Table + AsQuery, - T::Query: GroupByDsl, + T: Table + AsQuery>, + T::DefaultSelection: Expression + ValidGrouping<()>, + T::SqlType: TypedExpressionType, { - type Output = >::Output; + type Output = dsl::GroupBy, Expr>; - fn group_by(self, expr: Expr) -> Self::Output { + fn group_by(self, expr: Expr) -> dsl::GroupBy { self.as_query().group_by(expr) } } diff --git a/diesel/src/query_dsl/locking_dsl.rs b/diesel/src/query_dsl/locking_dsl.rs index 1c56d8f75f7f..c0de6e17f4ba 100644 --- a/diesel/src/query_dsl/locking_dsl.rs +++ b/diesel/src/query_dsl/locking_dsl.rs @@ -1,5 +1,9 @@ +use crate::expression::TypedExpressionType; +use crate::expression::ValidGrouping; use crate::query_builder::AsQuery; +use crate::query_builder::SelectStatement; use crate::query_source::Table; +use crate::Expression; /// Methods related to locking select statements /// @@ -21,10 +25,11 @@ pub trait LockingDsl { impl LockingDsl for T where - T: Table + AsQuery, - T::Query: LockingDsl, + T: Table + AsQuery>, + T::DefaultSelection: Expression + ValidGrouping<()>, + T::SqlType: TypedExpressionType, { - type Output = >::Output; + type Output = as LockingDsl>::Output; fn with_lock(self, lock: Lock) -> Self::Output { self.as_query().with_lock(lock) diff --git a/diesel/src/query_dsl/mod.rs b/diesel/src/query_dsl/mod.rs index 5433fdc7cf60..d3c45ff4744a 100644 --- a/diesel/src/query_dsl/mod.rs +++ b/diesel/src/query_dsl/mod.rs @@ -803,6 +803,7 @@ pub trait QueryDsl: Sized { /// let data = users::table.inner_join(posts::table) /// .group_by(users::id) /// .select((users::name, count(posts::id))) + /// # .order_by(users::id.asc()) /// .load::<(String, i64)>(&connection)?; /// /// assert_eq!(vec![(String::from("Sean"), 2), (String::from("Tess"), 1)], data); diff --git a/diesel_compile_tests/tests/compile-fail/boxed_queries_and_group_by.rs b/diesel_compile_tests/tests/compile-fail/boxed_queries_and_group_by.rs index f71821127750..030dd67069af 100644 --- a/diesel_compile_tests/tests/compile-fail/boxed_queries_and_group_by.rs +++ b/diesel_compile_tests/tests/compile-fail/boxed_queries_and_group_by.rs @@ -65,14 +65,6 @@ fn main() { .into_boxed(); //~^ ERROR BoxedDsl - // you cannot call group by after boxing - users::table - .into_boxed() - .group_by(users::id) - //~^ ERROR no method named `group_by` found - .select(users::name) - .load::(&conn); - users::table .group_by(users::name) .select(users::name) @@ -95,4 +87,14 @@ fn main() { // this is a different type now a = users::table.group_by(users::id).into_boxed(); //~^ ERROR mismatched types + + // you cannot call group by after boxing + users::table + .into_boxed() + .group_by(users::id) + //~^ ERROR type mismatch + //~| ERROR Table + //~| ERROR GroupByDsl + .select(users::name) + .load::(&conn); } diff --git a/diesel_compile_tests/tests/compile-fail/select_for_update_cannot_be_mixed_with_some_clauses.rs b/diesel_compile_tests/tests/compile-fail/select_for_update_cannot_be_mixed_with_some_clauses.rs index 1ec4db175a10..6a0db0b952a0 100644 --- a/diesel_compile_tests/tests/compile-fail/select_for_update_cannot_be_mixed_with_some_clauses.rs +++ b/diesel_compile_tests/tests/compile-fail/select_for_update_cannot_be_mixed_with_some_clauses.rs @@ -12,22 +12,35 @@ table! { fn main() { use self::users::dsl::*; - // FIXME: Overflows because of /~https://github.com/rust-lang/rust/issues/34260 - // should be E0277 - //users.for_update().distinct(); - // FIXME: Overflows because of /~https://github.com/rust-lang/rust/issues/34260 - // should be E0277 - // users.distinct().for_update(); + users.for_update().distinct(); + //~^ ERROR: E0271 + //~| ERROR: E0277 + //~| ERROR: E0277 + users.distinct().for_update(); + //~^ ERROR: E0271 + //~| ERROR: E0277 + users.for_update().distinct_on(id); + //~^ ERROR: E0271 + //~| ERROR: E0277 + //~| ERROR: E0277 + //~| ERROR: E0277 + users.distinct_on(id).for_update(); + //~^ ERROR: E0271 + //~| ERROR: E0277 users.for_update().group_by(id); - //~^ ERROR: E0599 - // FIXME: Overflows because of /~https://github.com/rust-lang/rust/issues/34260 - // should be E0277 - // users.group_by(id).for_update(); + //~^ ERROR: E0271 + //~| ERROR: E0277 + //~| ERROR: E0277 + users.group_by(id).for_update(); + //~^ ERROR: E0271 + //~| ERROR: E0277 - // FIXME: Overflows because of /~https://github.com/rust-lang/rust/issues/34260 - // should be E0277 - // users.into_boxed().for_update(); + users.into_boxed().for_update(); + //~^ ERROR: E0271 + //~| ERROR: E0277 users.for_update().into_boxed(); - //~^ ERROR: E0275 + //~^ ERROR: E0271 + //~| ERROR: E0277 + //~| ERROR: E0277 } From 0493edb44a1037cdc5016c6bfe8e25bc5c4ca94c Mon Sep 17 00:00:00 2001 From: Georg Semmler Date: Fri, 27 Nov 2020 09:39:40 +0100 Subject: [PATCH 4/4] Add a changelog entry --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 7e3512dccc17..fd0fc965d75e 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -50,6 +50,8 @@ for Rust libraries in [RFC #1105](/~https://github.com/rust-lang/rfcs/blob/master/ * Added `#[diesel(serialize_as)]` analogous to `#[diesel(deserialize_as)]`. This allows customization of the serialization behaviour of `Insertable` structs. +* Added support for `GROUP BY` clauses + ### Removed * All previously deprecated items have been removed.