Commit 2857ff3c authored by Vicențiu Ciorbaru's avatar Vicențiu Ciorbaru

MDEV-10815: Window Function Expressions Wrong Results

Fix window function expressions such as win_func() <operator> expr.
The problem was found in 2 places.
First, when we have complex expressions containing window functions, we
can only compute their final value _after_ we have computed the window
function's values. These values must be stored within the temporary
table that we are using, before sending them off.
This is done by performing an extra copy_funcs call before the final
end_send() call.

Second, such expressions need to have their inner arguments,
changed such that the references within those arguments point to fields within
the temporary table.
Ex: sum(t.a) over (order by t.b) + sum(t.a) over (order by t.b)
Before this fix, t.a pointed to the original table's a field. In order
to compute the sum function's value correctly, it needs to point to the
copy of this field inside the temp table.
This is done by calling split_sum_func for each argument in the
expression in turn.

The win.test results have also been updated as they contained wrong
values for such a use case.
parent 1c724413
...@@ -1816,15 +1816,15 @@ insert into t1 values ...@@ -1816,15 +1816,15 @@ insert into t1 values
(2,2), (2,2),
(3,1); (3,1);
select select
rank() over (order by a) - a, b,
rank() over (order by b) rank() over (order by a), rank() over (order by b),
rank() over (order by a) - rank() over (order by b) as diff
from from
t1; t1;
rank() over (order by a) - a b rank() over (order by a) rank() over (order by b) diff
rank() over (order by b) 1 3 1 3 -2
0 2 2 2 2 0
0 3 1 3 1 2
0
drop table t1; drop table t1;
create table t1 (i int); create table t1 (i int);
insert into t1 values (1),(2); insert into t1 values (1),(2);
...@@ -2055,7 +2055,7 @@ pk r_desc r_asc ...@@ -2055,7 +2055,7 @@ pk r_desc r_asc
11 1 11 11 1 11
drop table t1; drop table t1;
# #
# MDEV-10874: two window functions with ccompatible sorting # MDEV-10874: two window functions with compatible sorting
# #
create table t1 ( create table t1 (
pk int primary key, pk int primary key,
...@@ -2185,3 +2185,29 @@ EXPLAIN ...@@ -2185,3 +2185,29 @@ EXPLAIN
} }
} }
drop table t1; drop table t1;
#
# MDEV-10815: Window Function Expressions Wrong Results
#
create table t(a decimal(35,10), b int);
insert into t(a,b) values(1,1);
insert into t(a,b) values(2,1);
insert into t(a,b) values(0,1);
insert into t(a,b) values(1, 2);
insert into t(a,b) values(1.5,2);
insert into t(a,b) values(3, 2);
insert into t(a,b) values(4.5,2);
select a, b,
sum(t.a) over (partition by t.b order by a) as simple_sum,
sum(t.a) over (partition by t.b order by a) + 1 as sum_and_const,
sum(t.b) over (partition by t.b order by a) + sum(t.a) over (partition by t.b order by a) as sum_and_sum
from t
order by t.b, t.a;
a b simple_sum sum_and_const sum_and_sum
0.0000000000 1 0.0000000000 1.0000000000 1.0000000000
1.0000000000 1 1.0000000000 2.0000000000 3.0000000000
2.0000000000 1 3.0000000000 4.0000000000 6.0000000000
1.0000000000 2 1.0000000000 2.0000000000 3.0000000000
1.5000000000 2 2.5000000000 3.5000000000 6.5000000000
3.0000000000 2 5.5000000000 6.5000000000 11.5000000000
4.5000000000 2 10.0000000000 11.0000000000 18.0000000000
drop table t;
...@@ -1100,8 +1100,9 @@ insert into t1 values ...@@ -1100,8 +1100,9 @@ insert into t1 values
(3,1); (3,1);
select select
rank() over (order by a) - a, b,
rank() over (order by b) rank() over (order by a), rank() over (order by b),
rank() over (order by a) - rank() over (order by b) as diff
from from
t1; t1;
...@@ -1337,3 +1338,22 @@ select ...@@ -1337,3 +1338,22 @@ select
from t1; from t1;
drop table t1; drop table t1;
--echo #
--echo # MDEV-10815: Window Function Expressions Wrong Results
--echo #
create table t(a decimal(35,10), b int);
insert into t(a,b) values(1,1);
insert into t(a,b) values(2,1);
insert into t(a,b) values(0,1);
insert into t(a,b) values(1, 2);
insert into t(a,b) values(1.5,2);
insert into t(a,b) values(3, 2);
insert into t(a,b) values(4.5,2);
select a, b,
sum(t.a) over (partition by t.b order by a) as simple_sum,
sum(t.a) over (partition by t.b order by a) + 1 as sum_and_const,
sum(t.b) over (partition by t.b order by a) + sum(t.a) over (partition by t.b order by a) as sum_and_sum
from t
order by t.b, t.a;
drop table t;
...@@ -1876,7 +1876,11 @@ void Item::split_sum_func2(THD *thd, Ref_ptr_array ref_pointer_array, ...@@ -1876,7 +1876,11 @@ void Item::split_sum_func2(THD *thd, Ref_ptr_array ref_pointer_array,
Skip the else part, window functions are very special functions: Skip the else part, window functions are very special functions:
they need to have their own fields in the temp. table, but they they need to have their own fields in the temp. table, but they
need to be proceessed differently than regular aggregate functions need to be proceessed differently than regular aggregate functions
Call split_sum_func here so that each argument gets its fields to
point to the temporary table.
*/ */
split_sum_func(thd, ref_pointer_array, fields, split_flags);
} }
else else
{ {
......
...@@ -771,6 +771,7 @@ class Item_window_func : public Item_func_or_sum ...@@ -771,6 +771,7 @@ class Item_window_func : public Item_func_or_sum
void split_sum_func(THD *thd, Ref_ptr_array ref_pointer_array, void split_sum_func(THD *thd, Ref_ptr_array ref_pointer_array,
List<Item> &fields, uint flags); List<Item> &fields, uint flags);
void fix_length_and_dec() void fix_length_and_dec()
{ {
decimals = window_func()->decimals; decimals = window_func()->decimals;
......
...@@ -26228,8 +26228,23 @@ AGGR_OP::end_send() ...@@ -26228,8 +26228,23 @@ AGGR_OP::end_send()
rc= NESTED_LOOP_KILLED; rc= NESTED_LOOP_KILLED;
} }
else else
{
/*
In case we have window functions present, an extra step is required
to compute all the fields from the temporary table.
In case we have a compound expression such as: expr + expr,
where one of the terms has a window function inside it, only
after computing window function values we actually know the true
final result of the compounded expression.
Go through all the func items and save their values once again in the
corresponding temp table fields. Do this for each row in the table.
*/
if (join_tab->window_funcs_step)
copy_funcs(join_tab->tmp_table_param->items_to_copy, join->thd);
rc= evaluate_join_record(join, join_tab, 0); rc= evaluate_join_record(join, join_tab, 0);
} }
}
// Finish rnd scn after sending records // Finish rnd scn after sending records
if (join_tab->table->file->inited) if (join_tab->table->file->inited)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment