From 28dd9c2a304ec9df8cfdd86dfc50a616af8a0bed Mon Sep 17 00:00:00 2001 From: duripeng <453243496@qq.com> Date: Sat, 30 Apr 2022 04:34:25 +0800 Subject: [PATCH 1/5] sum(distinct) support --- datafusion/core/tests/sql/aggregates.rs | 57 ++++ .../physical-expr/src/aggregate/build_in.rs | 10 +- datafusion/physical-expr/src/aggregate/mod.rs | 1 + datafusion/physical-expr/src/aggregate/sum.rs | 12 + .../src/aggregate/sum_distinct.rs | 301 ++++++++++++++++++ .../physical-expr/src/expressions/mod.rs | 1 + 6 files changed, 377 insertions(+), 5 deletions(-) create mode 100644 datafusion/physical-expr/src/aggregate/sum_distinct.rs diff --git a/datafusion/core/tests/sql/aggregates.rs b/datafusion/core/tests/sql/aggregates.rs index 0569dcc09664..a5ae7d3479db 100644 --- a/datafusion/core/tests/sql/aggregates.rs +++ b/datafusion/core/tests/sql/aggregates.rs @@ -1236,6 +1236,63 @@ async fn simple_avg() -> Result<()> { Ok(()) } +#[tokio::test] +async fn query_sum_distinct() -> Result<()> { + let schema = Arc::new(Schema::new(vec![ + Field::new("c1", DataType::Int64, true), + Field::new("c2", DataType::Int64, true), + ])); + + let data = RecordBatch::try_new( + schema.clone(), + vec![ + Arc::new(Int64Array::from(vec![ + Some(0), + Some(1), + None, + Some(3), + Some(3), + ])), + Arc::new(Int64Array::from(vec![ + None, + Some(1), + Some(1), + Some(2), + Some(2), + ])), + ], + )?; + + let table = MemTable::try_new(schema, vec![vec![data]])?; + let ctx = SessionContext::new(); + ctx.register_table("test", Arc::new(table))?; + + // 2 different aggregate functions: avg and sum(distinct) + let sql = "SELECT AVG(c1), SUM(DISTINCT c2) FROM test"; + let actual = execute_to_batches(&ctx, sql).await; + let expected = vec![ + "+--------------+-----------------------+", + "| AVG(test.c1) | SUM(DISTINCT test.c2) |", + "+--------------+-----------------------+", + "| 1.75 | 3 |", + "+--------------+-----------------------+", + ]; + assert_batches_eq!(expected, &actual); + + // 2 sum(distinct) functions + let sql = "SELECT SUM(DISTINCT c1), SUM(DISTINCT c2) FROM test"; + let actual = execute_to_batches(&ctx, sql).await; + let expected = vec![ + "+-----------------------+-----------------------+", + "| SUM(DISTINCT test.c1) | SUM(DISTINCT test.c2) |", + "+-----------------------+-----------------------+", + "| 4 | 3 |", + "+-----------------------+-----------------------+", + ]; + assert_batches_eq!(expected, &actual); + Ok(()) +} + #[tokio::test] async fn query_count_distinct() -> Result<()> { let schema = Arc::new(Schema::new(vec![Field::new("c1", DataType::Int32, true)])); diff --git a/datafusion/physical-expr/src/aggregate/build_in.rs b/datafusion/physical-expr/src/aggregate/build_in.rs index f91e01336931..784cac81b1fb 100644 --- a/datafusion/physical-expr/src/aggregate/build_in.rs +++ b/datafusion/physical-expr/src/aggregate/build_in.rs @@ -87,11 +87,11 @@ pub fn create_aggregate_expr( name, return_type, )), - (AggregateFunction::Sum, true) => { - return Err(DataFusionError::NotImplemented( - "SUM(DISTINCT) aggregations are not available".to_string(), - )); - } + (AggregateFunction::Sum, true) => Arc::new(expressions::DistinctSum::new( + vec![coerced_phy_exprs[0].clone()], + name, + return_type, + )), (AggregateFunction::ApproxDistinct, _) => { Arc::new(expressions::ApproxDistinct::new( coerced_phy_exprs[0].clone(), diff --git a/datafusion/physical-expr/src/aggregate/mod.rs b/datafusion/physical-expr/src/aggregate/mod.rs index a9f3167c48d1..0ba160ffb2e6 100644 --- a/datafusion/physical-expr/src/aggregate/mod.rs +++ b/datafusion/physical-expr/src/aggregate/mod.rs @@ -42,6 +42,7 @@ mod hyperloglog; pub(crate) mod stats; pub(crate) mod stddev; pub(crate) mod sum; +pub(crate) mod sum_distinct; mod tdigest; pub(crate) mod variance; diff --git a/datafusion/physical-expr/src/aggregate/sum.rs b/datafusion/physical-expr/src/aggregate/sum.rs index 00404ee99b6f..2c9ce954d649 100644 --- a/datafusion/physical-expr/src/aggregate/sum.rs +++ b/datafusion/physical-expr/src/aggregate/sum.rs @@ -297,6 +297,18 @@ pub(crate) fn sum(lhs: &ScalarValue, rhs: &ScalarValue) -> Result { (ScalarValue::Int64(lhs), ScalarValue::Int8(rhs)) => { typed_sum!(lhs, rhs, Int64, i64) } + (ScalarValue::Int64(lhs), ScalarValue::UInt64(rhs)) => { + typed_sum!(lhs, rhs, Int64, i64) + } + (ScalarValue::Int64(lhs), ScalarValue::UInt32(rhs)) => { + typed_sum!(lhs, rhs, Int64, i64) + } + (ScalarValue::Int64(lhs), ScalarValue::UInt16(rhs)) => { + typed_sum!(lhs, rhs, Int64, i64) + } + (ScalarValue::Int64(lhs), ScalarValue::UInt8(rhs)) => { + typed_sum!(lhs, rhs, Int64, i64) + } e => { return Err(DataFusionError::Internal(format!( "Sum is not expected to receive a scalar {:?}", diff --git a/datafusion/physical-expr/src/aggregate/sum_distinct.rs b/datafusion/physical-expr/src/aggregate/sum_distinct.rs new file mode 100644 index 000000000000..c1a60779ffd6 --- /dev/null +++ b/datafusion/physical-expr/src/aggregate/sum_distinct.rs @@ -0,0 +1,301 @@ +// Licensed to the Apache Software Foundation (ASF) under one +// or more contributor license agreements. See the NOTICE file +// distributed with this work for additional information +// regarding copyright ownership. The ASF licenses this file +// to you under the Apache License, Version 2.0 (the +// "License"); you may not use this file except in compliance +// with the License. You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, +// software distributed under the License is distributed on an +// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +// KIND, either express or implied. See the License for the +// specific language governing permissions and limitations +// under the License. + +use crate::aggregate::sum; +use crate::expressions::format_state_name; +use arrow::datatypes::{DataType, Field}; +use std::any::Any; +use std::fmt::Debug; +use std::sync::Arc; + +use ahash::RandomState; +use arrow::array::{Array, ArrayRef}; +use std::collections::HashSet; + +use crate::{AggregateExpr, PhysicalExpr}; +use datafusion_common::ScalarValue; +use datafusion_common::{DataFusionError, Result}; +use datafusion_expr::Accumulator; + +/// Expression for a SUM(DISTINCT) aggregation. +#[derive(Debug)] +pub struct DistinctSum { + /// Column name + name: String, + /// The DataType for the final sum + data_type: DataType, + /// The input arguments, only contains 1 item for sum + exprs: Vec>, +} + +impl DistinctSum { + /// Create a SUM(DISTINCT) aggregate function. + pub fn new( + exprs: Vec>, + name: String, + data_type: DataType, + ) -> Self { + Self { + name, + data_type, + exprs, + } + } +} + +impl AggregateExpr for DistinctSum { + fn as_any(&self) -> &dyn Any { + self + } + + fn field(&self) -> Result { + Ok(Field::new(&self.name, self.data_type.clone(), true)) + } + + fn state_fields(&self) -> Result> { + // State field is a List which stores items to rebuild hash set. + Ok(vec![Field::new( + &format_state_name(&self.name, "sum distinct"), + DataType::List(Box::new(Field::new("item", self.data_type.clone(), true))), + false, + )]) + } + + fn expressions(&self) -> Vec> { + self.exprs.clone() + } + + fn name(&self) -> &str { + &self.name + } + + fn create_accumulator(&self) -> Result> { + Ok(Box::new(DistinctSumAccumulator::try_new(&self.data_type)?)) + } +} + +#[derive(Debug)] +struct DistinctSumAccumulator { + hash_values: HashSet, + data_type: DataType, +} +impl DistinctSumAccumulator { + pub fn try_new(data_type: &DataType) -> Result { + Ok(Self { + hash_values: HashSet::default(), + data_type: data_type.clone(), + }) + } + + fn update(&mut self, values: &[ScalarValue]) -> Result<()> { + values.into_iter().for_each(|v| { + // If the value is NULL, it is not included in the final sum. + if !v.is_null() { + self.hash_values.insert(v.clone()); + } + }); + + Ok(()) + } + + fn merge(&mut self, states: &[ScalarValue]) -> Result<()> { + if states.is_empty() { + return Ok(()); + } + + states.iter().try_for_each(|state| match state { + ScalarValue::List(Some(values), _) => self.update(values.as_ref()), + _ => Err(DataFusionError::Internal(format!( + "Unexpected accumulator state {:?}", + state + ))), + }) + } +} + +impl Accumulator for DistinctSumAccumulator { + fn state(&self) -> Result> { + let mut cols_out = { + let values = Box::new(Vec::new()); + let data_type = Box::new(self.data_type.clone()); + vec![ScalarValue::List(Some(values), data_type)] + }; + + let mut cols_vec = cols_out + .iter_mut() + .map(|c| match c { + ScalarValue::List(Some(ref mut v), _) => v, + _ => unreachable!(), + }) + .collect::>(); + + self.hash_values + .iter() + .for_each(|distinct_value| cols_vec[0].push(distinct_value.clone())); + Ok(cols_out) + } + + fn update_batch(&mut self, values: &[ArrayRef]) -> Result<()> { + if values.is_empty() { + return Ok(()); + } + + let scalar_values = (0..values[0].len()) + .map(|index| ScalarValue::try_from_array(&values[0], index)) + .collect::>>()?; + self.update(&scalar_values) + } + + fn merge_batch(&mut self, states: &[ArrayRef]) -> Result<()> { + if states.is_empty() { + return Ok(()); + } + + (0..states[0].len()).try_for_each(|index| { + let v = states + .iter() + .map(|array| ScalarValue::try_from_array(array, index)) + .collect::>>()?; + self.merge(&v) + }) + } + + fn evaluate(&self) -> Result { + let mut sum_value = ScalarValue::try_from(&self.data_type)?; + self.hash_values.iter().for_each(|distinct_value| { + sum_value = sum::sum(&sum_value, distinct_value).unwrap() + }); + Ok(sum_value) + } +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::expressions::col; + use crate::expressions::tests::aggregate; + use arrow::record_batch::RecordBatch; + use arrow::{array::*, datatypes::*}; + use datafusion_common::Result; + + fn run_update_batch( + return_type: DataType, + arrays: &[ArrayRef], + ) -> Result<(Vec, ScalarValue)> { + let agg = DistinctSum::new(vec![], String::from("__col_name__"), return_type); + + let mut accum = agg.create_accumulator()?; + accum.update_batch(arrays)?; + + Ok((accum.state()?, accum.evaluate()?)) + } + + macro_rules! generic_test_sum_distinct { + ($ARRAY:expr, $DATATYPE:expr, $EXPECTED:expr, $EXPECTED_DATATYPE:expr) => {{ + let schema = Schema::new(vec![Field::new("a", $DATATYPE, false)]); + + let batch = RecordBatch::try_new(Arc::new(schema.clone()), vec![$ARRAY])?; + + let agg = Arc::new(DistinctSum::new( + vec![col("a", &schema)?], + "count_distinct_a".to_string(), + $EXPECTED_DATATYPE, + )); + let actual = aggregate(&batch, agg)?; + let expected = ScalarValue::from($EXPECTED); + + assert_eq!(expected, actual); + + Ok(()) + }}; + } + + #[test] + fn sum_distinct_update_batch() -> Result<()> { + let array_int64: ArrayRef = Arc::new(Int64Array::from(vec![1, 1, 3])); + let arrays = vec![array_int64]; + let (states, result) = run_update_batch(DataType::Int64, &arrays)?; + + assert_eq!(states.len(), 1); + assert_eq!(result, ScalarValue::Int64(Some(4))); + + Ok(()) + } + + #[test] + fn sum_distinct_i32_with_nulls() -> Result<()> { + let array = Arc::new(Int32Array::from(vec![ + Some(1), + Some(1), + None, + Some(2), + Some(2), + Some(3), + ])); + generic_test_sum_distinct!( + array, + DataType::Int32, + ScalarValue::from(6i64), + DataType::Int64 + ) + } + + #[test] + fn sum_distinct_u32_with_nulls() -> Result<()> { + let array: ArrayRef = Arc::new(UInt32Array::from(vec![ + Some(1_u32), + Some(1_u32), + Some(3_u32), + Some(3_u32), + None, + ])); + generic_test_sum_distinct!( + array, + DataType::UInt32, + ScalarValue::from(4i64), + DataType::Int64 + ) + } + + #[test] + fn sum_distinct_f64() -> Result<()> { + let array: ArrayRef = + Arc::new(Float64Array::from(vec![1_f64, 1_f64, 3_f64, 3_f64, 3_f64])); + generic_test_sum_distinct!( + array, + DataType::Float64, + ScalarValue::from(4_f64), + DataType::Float64 + ) + } + + #[test] + fn sum_distinct_decimal_with_nulls() -> Result<()> { + let array: ArrayRef = Arc::new( + (1..6) + .map(|i| if i == 2 { None } else { Some(i % 2) }) + .collect::() + .with_precision_and_scale(35, 0)?, + ); + generic_test_sum_distinct!( + array, + DataType::Decimal(35, 0), + ScalarValue::Decimal128(Some(1), 38, 0), + DataType::Decimal(38, 0) + ) + } +} diff --git a/datafusion/physical-expr/src/expressions/mod.rs b/datafusion/physical-expr/src/expressions/mod.rs index 2cdceab6cc70..d081720b856d 100644 --- a/datafusion/physical-expr/src/expressions/mod.rs +++ b/datafusion/physical-expr/src/expressions/mod.rs @@ -55,6 +55,7 @@ pub use crate::aggregate::min_max::{MaxAccumulator, MinAccumulator}; pub use crate::aggregate::stats::StatsType; pub use crate::aggregate::stddev::{Stddev, StddevPop}; pub use crate::aggregate::sum::Sum; +pub use crate::aggregate::sum_distinct::DistinctSum; pub use crate::aggregate::variance::{Variance, VariancePop}; pub use crate::window::cume_dist::cume_dist; From bd3d7349e88c385f494c1397ba99cea67f2e583a Mon Sep 17 00:00:00 2001 From: duripeng <453243496@qq.com> Date: Mon, 2 May 2022 20:54:18 +0800 Subject: [PATCH 2/5] fix clippy --- datafusion/physical-expr/src/aggregate/sum_distinct.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/datafusion/physical-expr/src/aggregate/sum_distinct.rs b/datafusion/physical-expr/src/aggregate/sum_distinct.rs index c1a60779ffd6..6d4d5facb286 100644 --- a/datafusion/physical-expr/src/aggregate/sum_distinct.rs +++ b/datafusion/physical-expr/src/aggregate/sum_distinct.rs @@ -102,7 +102,7 @@ impl DistinctSumAccumulator { } fn update(&mut self, values: &[ScalarValue]) -> Result<()> { - values.into_iter().for_each(|v| { + values.iter().for_each(|v| { // If the value is NULL, it is not included in the final sum. if !v.is_null() { self.hash_values.insert(v.clone()); From 2936ea4f2ccf59c07374b62cbcca53c86126d1c4 Mon Sep 17 00:00:00 2001 From: duripeng <453243496@qq.com> Date: Tue, 3 May 2022 18:11:05 +0800 Subject: [PATCH 3/5] merge state() code logic --- .../src/aggregate/sum_distinct.rs | 25 +++++++------------ 1 file changed, 9 insertions(+), 16 deletions(-) diff --git a/datafusion/physical-expr/src/aggregate/sum_distinct.rs b/datafusion/physical-expr/src/aggregate/sum_distinct.rs index 6d4d5facb286..d48c2239de3c 100644 --- a/datafusion/physical-expr/src/aggregate/sum_distinct.rs +++ b/datafusion/physical-expr/src/aggregate/sum_distinct.rs @@ -129,24 +129,17 @@ impl DistinctSumAccumulator { impl Accumulator for DistinctSumAccumulator { fn state(&self) -> Result> { - let mut cols_out = { - let values = Box::new(Vec::new()); + // 1. Stores aggregate state in `ScalarValue::List` + // 2. Construct this `ScalarValue::List` state from distinct numeric in hash set + let state_out = { + let mut distinct_values = Box::new(Vec::new()); let data_type = Box::new(self.data_type.clone()); - vec![ScalarValue::List(Some(values), data_type)] + self.hash_values + .iter() + .for_each(|distinct_value| distinct_values.push(distinct_value.clone())); + vec![ScalarValue::List(Some(distinct_values), data_type)] }; - - let mut cols_vec = cols_out - .iter_mut() - .map(|c| match c { - ScalarValue::List(Some(ref mut v), _) => v, - _ => unreachable!(), - }) - .collect::>(); - - self.hash_values - .iter() - .for_each(|distinct_value| cols_vec[0].push(distinct_value.clone())); - Ok(cols_out) + Ok(state_out) } fn update_batch(&mut self, values: &[ArrayRef]) -> Result<()> { From 9fb347290922f54cb35fe3a271d417670c13ac5f Mon Sep 17 00:00:00 2001 From: duripeng <453243496@qq.com> Date: Tue, 3 May 2022 18:21:43 +0800 Subject: [PATCH 4/5] revise annotation --- datafusion/physical-expr/src/aggregate/sum_distinct.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/datafusion/physical-expr/src/aggregate/sum_distinct.rs b/datafusion/physical-expr/src/aggregate/sum_distinct.rs index d48c2239de3c..238722726547 100644 --- a/datafusion/physical-expr/src/aggregate/sum_distinct.rs +++ b/datafusion/physical-expr/src/aggregate/sum_distinct.rs @@ -130,7 +130,7 @@ impl DistinctSumAccumulator { impl Accumulator for DistinctSumAccumulator { fn state(&self) -> Result> { // 1. Stores aggregate state in `ScalarValue::List` - // 2. Construct this `ScalarValue::List` state from distinct numeric in hash set + // 2. Constructs `ScalarValue::List` state from distinct numeric stored in hash set let state_out = { let mut distinct_values = Box::new(Vec::new()); let data_type = Box::new(self.data_type.clone()); From cfacb16256b5af8d793305b4dfc7d7224190a230 Mon Sep 17 00:00:00 2001 From: duripeng <453243496@qq.com> Date: Wed, 4 May 2022 01:37:26 +0800 Subject: [PATCH 5/5] remove u64->i63 coercion --- datafusion/physical-expr/src/aggregate/sum.rs | 3 --- 1 file changed, 3 deletions(-) diff --git a/datafusion/physical-expr/src/aggregate/sum.rs b/datafusion/physical-expr/src/aggregate/sum.rs index 2c9ce954d649..cca54733d246 100644 --- a/datafusion/physical-expr/src/aggregate/sum.rs +++ b/datafusion/physical-expr/src/aggregate/sum.rs @@ -297,9 +297,6 @@ pub(crate) fn sum(lhs: &ScalarValue, rhs: &ScalarValue) -> Result { (ScalarValue::Int64(lhs), ScalarValue::Int8(rhs)) => { typed_sum!(lhs, rhs, Int64, i64) } - (ScalarValue::Int64(lhs), ScalarValue::UInt64(rhs)) => { - typed_sum!(lhs, rhs, Int64, i64) - } (ScalarValue::Int64(lhs), ScalarValue::UInt32(rhs)) => { typed_sum!(lhs, rhs, Int64, i64) }