Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
taosdata
TDengine
提交
a6e7f0f2
T
TDengine
项目概览
taosdata
/
TDengine
1 年多 前同步成功
通知
1185
Star
22016
Fork
4786
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
1
列表
看板
标记
里程碑
合并请求
0
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
T
TDengine
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
1
Issue
1
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
未验证
提交
a6e7f0f2
编写于
12月 25, 2022
作者:
D
dapan1121
提交者:
GitHub
12月 25, 2022
浏览文件
操作
浏览文件
下载
差异文件
Merge pull request #18960 from taosdata/fix/TD-20621-3.0
fix(query): avg function support overflow sum value calculate
上级
a2ae3515
0a5fd77b
变更
3
显示空白变更内容
内联
并排
Showing
3 changed file
with
128 addition
and
25 deletion
+128
-25
source/libs/function/inc/builtinsimpl.h
source/libs/function/inc/builtinsimpl.h
+1
-0
source/libs/function/src/detail/tavgfunction.c
source/libs/function/src/detail/tavgfunction.c
+80
-23
tests/system-test/2-query/avg.py
tests/system-test/2-query/avg.py
+47
-2
未找到文件。
source/libs/function/inc/builtinsimpl.h
浏览文件 @
a6e7f0f2
...
...
@@ -32,6 +32,7 @@ typedef struct SSumRes {
int16_t
type
;
int64_t
prevTs
;
bool
isPrevTsSet
;
bool
overflow
;
// if overflow is true, dsum to be used for any type;
}
SSumRes
;
typedef
struct
SMinmaxResInfo
{
...
...
source/libs/function/src/detail/tavgfunction.c
浏览文件 @
a6e7f0f2
...
...
@@ -41,6 +41,57 @@
} \
} while (0)
// define signed number sum with check overflow
#define CHECK_OVERFLOW_SUM_SIGNED(out, val) \
if (out->sum.overflow) { \
out->sum.dsum += val; \
} else if (out->sum.isum > 0 && val > 0 && INT64_MAX - out->sum.isum <= val || \
out->sum.isum < 0 && val < 0 && INT64_MIN - out->sum.isum >= val) { \
double dsum = (double)out->sum.isum; \
out->sum.overflow = true; \
out->sum.dsum = dsum + val; \
} else { \
out->sum.isum += val; \
}
// val is big than INT64_MAX, val come from merge
#define CHECK_OVERFLOW_SUM_SIGNED_BIG(out, val, big) \
if (out->sum.overflow) { \
out->sum.dsum += val; \
} else if (out->sum.isum > 0 && val > 0 && INT64_MAX - out->sum.isum <= val || \
out->sum.isum < 0 && val < 0 && INT64_MIN - out->sum.isum >= val || \
big) { \
double dsum = (double)out->sum.isum; \
out->sum.overflow = true; \
out->sum.dsum = dsum + val; \
} else { \
out->sum.isum += val; \
}
// define unsigned number sum with check overflow
#define CHECK_OVERFLOW_SUM_UNSIGNED(out, val) \
if (out->sum.overflow) { \
out->sum.dsum += val; \
} else if (UINT64_MAX - out->sum.usum <= val) { \
double dsum = (double)out->sum.usum; \
out->sum.overflow = true; \
out->sum.dsum = dsum + val; \
} else { \
out->sum.usum += val; \
}
// val is big than UINT64_MAX, val come from merge
#define CHECK_OVERFLOW_SUM_UNSIGNED_BIG(out, val, big) \
if (out->sum.overflow) { \
out->sum.dsum += val; \
} else if (UINT64_MAX - out->sum.usum <= val || big) { \
double dsum = (double)out->sum.usum; \
out->sum.overflow = true; \
out->sum.dsum = dsum + val; \
} else { \
out->sum.usum += val; \
}
typedef
struct
SAvgRes
{
double
result
;
SSumRes
sum
;
...
...
@@ -319,9 +370,9 @@ static int32_t calculateAvgBySMAInfo(SAvgRes* pRes, int32_t numOfRows, int32_t t
pRes
->
count
+=
numOfElem
;
if
(
IS_SIGNED_NUMERIC_TYPE
(
type
))
{
pRes
->
sum
.
isum
+=
pAgg
->
sum
;
CHECK_OVERFLOW_SUM_SIGNED
(
pRes
,
pAgg
->
sum
)
;
}
else
if
(
IS_UNSIGNED_NUMERIC_TYPE
(
type
))
{
pRes
->
sum
.
usum
+=
pAgg
->
sum
;
CHECK_OVERFLOW_SUM_UNSIGNED
(
pRes
,
pAgg
->
sum
)
;
}
else
if
(
IS_FLOAT_TYPE
(
type
))
{
pRes
->
sum
.
dsum
+=
GET_DOUBLE_VAL
((
const
char
*
)
&
(
pAgg
->
sum
));
}
...
...
@@ -344,7 +395,7 @@ static int32_t doAddNumericVector(SColumnInfoData* pCol, int32_t type, SInputCol
numOfElems
+=
1
;
pRes
->
count
+=
1
;
pRes
->
sum
.
isum
+=
plist
[
i
];
CHECK_OVERFLOW_SUM_SIGNED
(
pRes
,
plist
[
i
])
}
break
;
...
...
@@ -359,7 +410,7 @@ static int32_t doAddNumericVector(SColumnInfoData* pCol, int32_t type, SInputCol
numOfElems
+=
1
;
pRes
->
count
+=
1
;
pRes
->
sum
.
isum
+=
plist
[
i
];
CHECK_OVERFLOW_SUM_SIGNED
(
pRes
,
plist
[
i
])
}
break
;
}
...
...
@@ -373,7 +424,7 @@ static int32_t doAddNumericVector(SColumnInfoData* pCol, int32_t type, SInputCol
numOfElems
+=
1
;
pRes
->
count
+=
1
;
pRes
->
sum
.
isum
+=
plist
[
i
];
CHECK_OVERFLOW_SUM_SIGNED
(
pRes
,
plist
[
i
])
}
break
;
...
...
@@ -388,7 +439,7 @@ static int32_t doAddNumericVector(SColumnInfoData* pCol, int32_t type, SInputCol
numOfElems
+=
1
;
pRes
->
count
+=
1
;
pRes
->
sum
.
isum
+=
plist
[
i
];
CHECK_OVERFLOW_SUM_SIGNED
(
pRes
,
plist
[
i
])
}
break
;
}
...
...
@@ -402,7 +453,7 @@ static int32_t doAddNumericVector(SColumnInfoData* pCol, int32_t type, SInputCol
numOfElems
+=
1
;
pRes
->
count
+=
1
;
pRes
->
sum
.
usum
+=
plist
[
i
];
CHECK_OVERFLOW_SUM_UNSIGNED
(
pRes
,
plist
[
i
])
}
break
;
...
...
@@ -417,7 +468,7 @@ static int32_t doAddNumericVector(SColumnInfoData* pCol, int32_t type, SInputCol
numOfElems
+=
1
;
pRes
->
count
+=
1
;
pRes
->
sum
.
usum
+=
plist
[
i
];
CHECK_OVERFLOW_SUM_UNSIGNED
(
pRes
,
plist
[
i
])
}
break
;
}
...
...
@@ -431,7 +482,7 @@ static int32_t doAddNumericVector(SColumnInfoData* pCol, int32_t type, SInputCol
numOfElems
+=
1
;
pRes
->
count
+=
1
;
pRes
->
sum
.
usum
+=
plist
[
i
];
CHECK_OVERFLOW_SUM_UNSIGNED
(
pRes
,
plist
[
i
])
}
break
;
...
...
@@ -446,7 +497,8 @@ static int32_t doAddNumericVector(SColumnInfoData* pCol, int32_t type, SInputCol
numOfElems
+=
1
;
pRes
->
count
+=
1
;
pRes
->
sum
.
usum
+=
plist
[
i
];
CHECK_OVERFLOW_SUM_UNSIGNED
(
pRes
,
plist
[
i
])
}
break
;
}
...
...
@@ -527,9 +579,9 @@ int32_t avgFunction(SqlFunctionCtx* pCtx) {
}
else
{
for
(
int32_t
i
=
pInput
->
startRowIndex
;
i
<
pInput
->
numOfRows
+
pInput
->
startRowIndex
;
++
i
)
{
if
(
type
==
TSDB_DATA_TYPE_TINYINT
)
{
pAvgRes
->
sum
.
isum
+=
plist
[
i
];
CHECK_OVERFLOW_SUM_SIGNED
(
pAvgRes
,
plist
[
i
])
}
else
{
pAvgRes
->
sum
.
usum
+=
(
uint8_t
)
plist
[
i
];
CHECK_OVERFLOW_SUM_UNSIGNED
(
pAvgRes
,
(
uint8_t
)
plist
[
i
])
}
}
}
...
...
@@ -546,9 +598,9 @@ int32_t avgFunction(SqlFunctionCtx* pCtx) {
}
else
{
for
(
int32_t
i
=
pInput
->
startRowIndex
;
i
<
pInput
->
numOfRows
+
pInput
->
startRowIndex
;
++
i
)
{
if
(
type
==
TSDB_DATA_TYPE_SMALLINT
)
{
pAvgRes
->
sum
.
isum
+=
plist
[
i
];
CHECK_OVERFLOW_SUM_SIGNED
(
pAvgRes
,
plist
[
i
])
}
else
{
pAvgRes
->
sum
.
usum
+=
(
uint16_t
)
plist
[
i
];
CHECK_OVERFLOW_SUM_UNSIGNED
(
pAvgRes
,
(
uint16_t
)
plist
[
i
])
}
}
}
...
...
@@ -565,9 +617,9 @@ int32_t avgFunction(SqlFunctionCtx* pCtx) {
}
else
{
for
(
int32_t
i
=
pInput
->
startRowIndex
;
i
<
pInput
->
numOfRows
+
pInput
->
startRowIndex
;
++
i
)
{
if
(
type
==
TSDB_DATA_TYPE_INT
)
{
pAvgRes
->
sum
.
isum
+=
plist
[
i
];
CHECK_OVERFLOW_SUM_SIGNED
(
pAvgRes
,
plist
[
i
])
}
else
{
pAvgRes
->
sum
.
usum
+=
(
uint32_t
)
plist
[
i
];
CHECK_OVERFLOW_SUM_UNSIGNED
(
pAvgRes
,
(
uint32_t
)
plist
[
i
])
}
}
}
...
...
@@ -584,9 +636,9 @@ int32_t avgFunction(SqlFunctionCtx* pCtx) {
}
else
{
for
(
int32_t
i
=
pInput
->
startRowIndex
;
i
<
pInput
->
numOfRows
+
pInput
->
startRowIndex
;
++
i
)
{
if
(
type
==
TSDB_DATA_TYPE_BIGINT
)
{
pAvgRes
->
sum
.
isum
+=
plist
[
i
];
CHECK_OVERFLOW_SUM_SIGNED
(
pAvgRes
,
plist
[
i
])
}
else
{
pAvgRes
->
sum
.
usum
+=
(
uint64_t
)
plist
[
i
];
CHECK_OVERFLOW_SUM_UNSIGNED
(
pAvgRes
,
(
uint64_t
)
plist
[
i
])
}
}
}
...
...
@@ -639,9 +691,11 @@ static void avgTransferInfo(SAvgRes* pInput, SAvgRes* pOutput) {
pOutput
->
type
=
pInput
->
type
;
if
(
IS_SIGNED_NUMERIC_TYPE
(
pOutput
->
type
))
{
pOutput
->
sum
.
isum
+=
pInput
->
sum
.
isum
;
bool
overflow
=
pInput
->
sum
.
overflow
;
CHECK_OVERFLOW_SUM_SIGNED_BIG
(
pOutput
,
(
overflow
?
pInput
->
sum
.
dsum
:
pInput
->
sum
.
isum
),
overflow
);
}
else
if
(
IS_UNSIGNED_NUMERIC_TYPE
(
pOutput
->
type
))
{
pOutput
->
sum
.
usum
+=
pInput
->
sum
.
usum
;
bool
overflow
=
pInput
->
sum
.
overflow
;
CHECK_OVERFLOW_SUM_UNSIGNED_BIG
(
pOutput
,
(
overflow
?
pInput
->
sum
.
dsum
:
pInput
->
sum
.
usum
),
overflow
);
}
else
{
pOutput
->
sum
.
dsum
+=
pInput
->
sum
.
dsum
;
}
...
...
@@ -741,9 +795,9 @@ int32_t avgCombine(SqlFunctionCtx* pDestCtx, SqlFunctionCtx* pSourceCtx) {
int16_t
type
=
pDBuf
->
type
==
TSDB_DATA_TYPE_NULL
?
pSBuf
->
type
:
pDBuf
->
type
;
if
(
IS_SIGNED_NUMERIC_TYPE
(
type
))
{
pDBuf
->
sum
.
isum
+=
pSBuf
->
sum
.
isum
;
CHECK_OVERFLOW_SUM_SIGNED
(
pDBuf
,
pSBuf
->
sum
.
isum
)
}
else
if
(
IS_UNSIGNED_NUMERIC_TYPE
(
type
))
{
pDBuf
->
sum
.
usum
+=
pSBuf
->
sum
.
usum
;
CHECK_OVERFLOW_SUM_UNSIGNED
(
pDBuf
,
pSBuf
->
sum
.
usum
)
}
else
{
pDBuf
->
sum
.
dsum
+=
pSBuf
->
sum
.
dsum
;
}
...
...
@@ -759,7 +813,10 @@ int32_t avgFinalize(SqlFunctionCtx* pCtx, SSDataBlock* pBlock) {
int32_t
type
=
pRes
->
type
;
if
(
pRes
->
count
>
0
)
{
if
(
IS_SIGNED_NUMERIC_TYPE
(
type
))
{
if
(
pRes
->
sum
.
overflow
)
{
// overflow flag set , use dsum
pRes
->
result
=
pRes
->
sum
.
dsum
/
((
double
)
pRes
->
count
);
}
else
if
(
IS_SIGNED_NUMERIC_TYPE
(
type
))
{
pRes
->
result
=
pRes
->
sum
.
isum
/
((
double
)
pRes
->
count
);
}
else
if
(
IS_UNSIGNED_NUMERIC_TYPE
(
type
))
{
pRes
->
result
=
pRes
->
sum
.
usum
/
((
double
)
pRes
->
count
);
...
...
tests/system-test/2-query/avg.py
浏览文件 @
a6e7f0f2
...
...
@@ -409,7 +409,7 @@ class TDTestCase:
tdSql
.
query
(
f
"select avg(c1) ,avg(c2) , avg(c3) , avg(c4), avg(c5), avg(c6) from
{
dbname
}
.sub1_bound "
)
tdSql
.
checkRows
(
1
)
tdSql
.
checkData
(
0
,
0
,
920350133.571428537
)
tdSql
.
checkData
(
0
,
1
,
1.3176245766935393
e+18
)
tdSql
.
checkData
(
0
,
1
,
3.952873730080618
e+18
)
tdSql
.
checkData
(
0
,
2
,
14042.142857143
)
tdSql
.
checkData
(
0
,
3
,
53.571428571
)
tdSql
.
checkData
(
0
,
4
,
5.828571332045761e+37
)
...
...
@@ -419,13 +419,56 @@ class TDTestCase:
# check + - * / in functions
tdSql
.
query
(
f
" select avg(c1+1) ,avg(c2) , avg(c3*1) , avg(c4/2), avg(c5)/2, avg(c6) from
{
dbname
}
.sub1_bound "
)
tdSql
.
checkData
(
0
,
0
,
920350134.5714285
)
tdSql
.
checkData
(
0
,
1
,
1.3176245766935393
e+18
)
tdSql
.
checkData
(
0
,
1
,
3.952873730080618
e+18
)
tdSql
.
checkData
(
0
,
2
,
14042.142857143
)
tdSql
.
checkData
(
0
,
3
,
26.785714286
)
tdSql
.
checkData
(
0
,
4
,
2.9142856660228804e+37
)
tdSql
.
checkData
(
0
,
5
,
None
)
#
# test bigint to check overflow
#
def
avg_check_overflow
(
self
):
# create db
tdSql
.
execute
(
f
"drop database if exists db"
)
tdSql
.
execute
(
f
"create database if not exists db"
)
time
.
sleep
(
3
)
tdSql
.
execute
(
f
"use db"
)
tdSql
.
execute
(
f
"create table db.st(ts timestamp, ibv bigint, ubv bigint unsigned) tags(area int)"
)
# insert t1 data
tdSql
.
execute
(
f
"insert into db.t1 using db.st tags(1) values(now,9223372036854775801,18446744073709551611)"
)
tdSql
.
execute
(
f
"insert into db.t1 using db.st tags(1) values(now,8223372036854775801,17446744073709551611)"
)
tdSql
.
execute
(
f
"insert into db.t1 using db.st tags(1) values(now,7223372036854775801,16446744073709551611)"
)
# insert t2 data
tdSql
.
execute
(
f
"insert into db.t2 using db.st tags(2) values(now,9223372036854775801,18446744073709551611)"
)
tdSql
.
execute
(
f
"insert into db.t2 using db.st tags(2) values(now,8223372036854775801,17446744073709551611)"
)
tdSql
.
execute
(
f
"insert into db.t2 using db.st tags(2) values(now,7223372036854775801,16446744073709551611)"
)
# check single table answer
tdSql
.
query
(
f
"select avg(ibv), avg(ubv) from db.t1"
)
tdSql
.
checkRows
(
1
)
tdSql
.
checkData
(
0
,
0
,
8.223372036854776e+18
)
tdSql
.
checkData
(
0
,
1
,
1.744674407370955e+19
)
# check super table
tdSql
.
query
(
f
"select avg(ibv), avg(ubv) from db.st"
)
tdSql
.
checkRows
(
1
)
tdSql
.
checkData
(
0
,
0
,
8.223372036854776e+18
)
tdSql
.
checkData
(
0
,
1
,
1.744674407370955e+19
)
# check child query
tdSql
.
query
(
f
"select avg(ibv), avg(ubv) from (select * from db.st)"
)
tdSql
.
checkRows
(
1
)
tdSql
.
checkData
(
0
,
0
,
8.223372036854776e+18
)
tdSql
.
checkData
(
0
,
1
,
1.744674407370955e+19
)
# check group by
tdSql
.
query
(
f
"select avg(ibv), avg(ubv) from db.st group by tbname"
)
tdSql
.
checkRows
(
2
)
tdSql
.
checkData
(
0
,
0
,
8.223372036854776e+18
)
tdSql
.
checkData
(
0
,
1
,
1.744674407370955e+19
)
tdSql
.
checkData
(
1
,
0
,
8.223372036854776e+18
)
tdSql
.
checkData
(
1
,
1
,
1.744674407370955e+19
)
def
run
(
self
):
# sourcery skip: extract-duplicate-method, remove-redundant-fstring
tdSql
.
prepare
()
...
...
@@ -455,6 +498,8 @@ class TDTestCase:
self
.
avg_func_filter
()
self
.
avg_check_unsigned
()
# check avg overflow
self
.
avg_check_overflow
()
def
stop
(
self
):
tdSql
.
close
()
tdLog
.
success
(
f
"
{
__file__
}
successfully executed"
)
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录