mirror of
https://github.com/MariaDB/server.git
synced 2025-08-08 11:22:35 +03:00
ANALYZE FORMAT=JSON: better output and tests
- Print r_loops - Always print r_* members. Print NULL values if no scans took place - Added testcases.
This commit is contained in:
177
mysql-test/r/analyze_format_json.result
Normal file
177
mysql-test/r/analyze_format_json.result
Normal file
@@ -0,0 +1,177 @@
|
|||||||
|
drop table if exists t0,t1,t2,t3;
|
||||||
|
create table t0 (a int);
|
||||||
|
INSERT INTO t0 VALUES (0),(1),(2),(3),(4),(5),(6),(7),(8),(9);
|
||||||
|
# r_filtered=30%, because 3 rows match: 0,1,2
|
||||||
|
analyze format=json select * from t0 where a<3;
|
||||||
|
EXPLAIN
|
||||||
|
{
|
||||||
|
"query_block": {
|
||||||
|
"select_id": 1,
|
||||||
|
"table": {
|
||||||
|
"table_name": "t0",
|
||||||
|
"access_type": "ALL",
|
||||||
|
"r_loops": 1,
|
||||||
|
"rows": 10,
|
||||||
|
"r_rows": 10,
|
||||||
|
"filtered": 100,
|
||||||
|
"r_filtered": 30,
|
||||||
|
"attached_condition": "(t0.a < 3)"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
create table t1 (a int, b int, c int, key(a));
|
||||||
|
insert into t1 select A.a*10 + B.a, A.a*10 + B.a, A.a*10 + B.a from t0 A, t0 B;
|
||||||
|
analyze
|
||||||
|
select * from t0, t1 where t1.a=t0.a and t0.a > 9;
|
||||||
|
id select_type table type possible_keys key key_len ref rows r_rows filtered r_filtered Extra
|
||||||
|
1 SIMPLE t0 ALL NULL NULL NULL NULL 10 10 100.00 0.00 Using where
|
||||||
|
1 SIMPLE t1 ref a a 5 test.t0.a 1 NULL 100.00 NULL
|
||||||
|
analyze format=json
|
||||||
|
select * from t0, t1 where t1.a=t0.a and t0.a > 9;
|
||||||
|
EXPLAIN
|
||||||
|
{
|
||||||
|
"query_block": {
|
||||||
|
"select_id": 1,
|
||||||
|
"table": {
|
||||||
|
"table_name": "t0",
|
||||||
|
"access_type": "ALL",
|
||||||
|
"r_loops": 1,
|
||||||
|
"rows": 10,
|
||||||
|
"r_rows": 10,
|
||||||
|
"filtered": 100,
|
||||||
|
"r_filtered": 0,
|
||||||
|
"attached_condition": "((t0.a > 9) and (t0.a is not null))"
|
||||||
|
},
|
||||||
|
"table": {
|
||||||
|
"table_name": "t1",
|
||||||
|
"access_type": "ref",
|
||||||
|
"possible_keys": ["a"],
|
||||||
|
"key": "a",
|
||||||
|
"key_length": "5",
|
||||||
|
"used_key_parts": ["a"],
|
||||||
|
"ref": ["test.t0.a"],
|
||||||
|
"r_loops": 0,
|
||||||
|
"rows": 1,
|
||||||
|
"r_rows": null,
|
||||||
|
"filtered": 100,
|
||||||
|
"r_filtered": null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
analyze
|
||||||
|
select * from t0, t1 where t1.a=t0.a and t1.b<4;
|
||||||
|
id select_type table type possible_keys key key_len ref rows r_rows filtered r_filtered Extra
|
||||||
|
1 SIMPLE t0 ALL NULL NULL NULL NULL 10 10 100.00 100.00 Using where
|
||||||
|
1 SIMPLE t1 ref a a 5 test.t0.a 1 1 100.00 40.00 Using where
|
||||||
|
analyze format=json
|
||||||
|
select * from t0, t1 where t1.a=t0.a and t1.b<4;
|
||||||
|
EXPLAIN
|
||||||
|
{
|
||||||
|
"query_block": {
|
||||||
|
"select_id": 1,
|
||||||
|
"table": {
|
||||||
|
"table_name": "t0",
|
||||||
|
"access_type": "ALL",
|
||||||
|
"r_loops": 1,
|
||||||
|
"rows": 10,
|
||||||
|
"r_rows": 10,
|
||||||
|
"filtered": 100,
|
||||||
|
"r_filtered": 100,
|
||||||
|
"attached_condition": "(t0.a is not null)"
|
||||||
|
},
|
||||||
|
"table": {
|
||||||
|
"table_name": "t1",
|
||||||
|
"access_type": "ref",
|
||||||
|
"possible_keys": ["a"],
|
||||||
|
"key": "a",
|
||||||
|
"key_length": "5",
|
||||||
|
"used_key_parts": ["a"],
|
||||||
|
"ref": ["test.t0.a"],
|
||||||
|
"r_loops": 10,
|
||||||
|
"rows": 1,
|
||||||
|
"r_rows": 1,
|
||||||
|
"filtered": 100,
|
||||||
|
"r_filtered": 40,
|
||||||
|
"attached_condition": "(t1.b < 4)"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
analyze
|
||||||
|
select * from t1 A, t1 B where A.b<2 and B.b>5;
|
||||||
|
id select_type table type possible_keys key key_len ref rows r_rows filtered r_filtered Extra
|
||||||
|
1 SIMPLE A ALL NULL NULL NULL NULL 100 100 100.00 2.00 Using where
|
||||||
|
1 SIMPLE B ALL NULL NULL NULL NULL 100 100 100.00 94.00 Using where; Using join buffer (flat, BNL join)
|
||||||
|
analyze format=json
|
||||||
|
select * from t1 A, t1 B where A.b<20 and B.b<60;
|
||||||
|
EXPLAIN
|
||||||
|
{
|
||||||
|
"query_block": {
|
||||||
|
"select_id": 1,
|
||||||
|
"table": {
|
||||||
|
"table_name": "A",
|
||||||
|
"access_type": "ALL",
|
||||||
|
"r_loops": 1,
|
||||||
|
"rows": 100,
|
||||||
|
"r_rows": 100,
|
||||||
|
"filtered": 100,
|
||||||
|
"r_filtered": 20,
|
||||||
|
"attached_condition": "(A.b < 20)"
|
||||||
|
},
|
||||||
|
"block-nl-join": {
|
||||||
|
"table": {
|
||||||
|
"table_name": "B",
|
||||||
|
"access_type": "ALL",
|
||||||
|
"r_loops": 1,
|
||||||
|
"rows": 100,
|
||||||
|
"r_rows": 100,
|
||||||
|
"filtered": 100,
|
||||||
|
"r_filtered": 60
|
||||||
|
},
|
||||||
|
"buffer_type": "flat",
|
||||||
|
"join_type": "BNL",
|
||||||
|
"r_filtered": 100
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
analyze format=json
|
||||||
|
select * from t1 A, t1 B where A.b<20 and B.b<60 and A.c > B.c;
|
||||||
|
EXPLAIN
|
||||||
|
{
|
||||||
|
"query_block": {
|
||||||
|
"select_id": 1,
|
||||||
|
"table": {
|
||||||
|
"table_name": "A",
|
||||||
|
"access_type": "ALL",
|
||||||
|
"r_loops": 1,
|
||||||
|
"rows": 100,
|
||||||
|
"r_rows": 100,
|
||||||
|
"filtered": 100,
|
||||||
|
"r_filtered": 20,
|
||||||
|
"attached_condition": "(A.b < 20)"
|
||||||
|
},
|
||||||
|
"block-nl-join": {
|
||||||
|
"table": {
|
||||||
|
"table_name": "B",
|
||||||
|
"access_type": "ALL",
|
||||||
|
"r_loops": 1,
|
||||||
|
"rows": 100,
|
||||||
|
"r_rows": 100,
|
||||||
|
"filtered": 100,
|
||||||
|
"r_filtered": 60,
|
||||||
|
"attached_condition": "(B.b < 60)"
|
||||||
|
},
|
||||||
|
"buffer_type": "flat",
|
||||||
|
"join_type": "BNL",
|
||||||
|
"attached_condition": "(A.c > B.c)",
|
||||||
|
"r_filtered": 15.833
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
select count(*) from t1 A, t1 B where A.b<20 and B.b<60;
|
||||||
|
count(*)
|
||||||
|
1200
|
||||||
|
select count(*) from t1 A, t1 B where A.b<20 and B.b<60 and A.c > B.c;
|
||||||
|
count(*)
|
||||||
|
190
|
||||||
|
drop table t1;
|
||||||
|
drop table t0;
|
38
mysql-test/t/analyze_format_json.test
Normal file
38
mysql-test/t/analyze_format_json.test
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
#
|
||||||
|
# Tests for "ANALYZE FORMAT=JSON $statement" syntax
|
||||||
|
#
|
||||||
|
--disable_warnings
|
||||||
|
drop table if exists t0,t1,t2,t3;
|
||||||
|
--enable_warnings
|
||||||
|
|
||||||
|
create table t0 (a int);
|
||||||
|
INSERT INTO t0 VALUES (0),(1),(2),(3),(4),(5),(6),(7),(8),(9);
|
||||||
|
|
||||||
|
--echo # r_filtered=30%, because 3 rows match: 0,1,2
|
||||||
|
analyze format=json select * from t0 where a<3;
|
||||||
|
|
||||||
|
create table t1 (a int, b int, c int, key(a));
|
||||||
|
insert into t1 select A.a*10 + B.a, A.a*10 + B.a, A.a*10 + B.a from t0 A, t0 B;
|
||||||
|
|
||||||
|
analyze
|
||||||
|
select * from t0, t1 where t1.a=t0.a and t0.a > 9;
|
||||||
|
analyze format=json
|
||||||
|
select * from t0, t1 where t1.a=t0.a and t0.a > 9;
|
||||||
|
|
||||||
|
analyze
|
||||||
|
select * from t0, t1 where t1.a=t0.a and t1.b<4;
|
||||||
|
|
||||||
|
analyze format=json
|
||||||
|
select * from t0, t1 where t1.a=t0.a and t1.b<4;
|
||||||
|
|
||||||
|
analyze
|
||||||
|
select * from t1 A, t1 B where A.b<2 and B.b>5;
|
||||||
|
|
||||||
|
analyze format=json
|
||||||
|
select * from t1 A, t1 B where A.b<20 and B.b<60;
|
||||||
|
|
||||||
|
analyze format=json
|
||||||
|
select * from t1 A, t1 B where A.b<20 and B.b<60 and A.c > B.c;
|
||||||
|
|
||||||
|
drop table t1;
|
||||||
|
drop table t0;
|
@@ -131,6 +131,12 @@ void Json_writer::add_bool(bool val)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Json_writer::add_null()
|
||||||
|
{
|
||||||
|
add_unquoted_str("null");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
void Json_writer::add_unquoted_str(const char* str)
|
void Json_writer::add_unquoted_str(const char* str)
|
||||||
{
|
{
|
||||||
if (fmt_helper.on_add_str(str))
|
if (fmt_helper.on_add_str(str))
|
||||||
|
@@ -99,6 +99,7 @@ public:
|
|||||||
void add_ll(longlong val);
|
void add_ll(longlong val);
|
||||||
void add_double(double val);
|
void add_double(double val);
|
||||||
void add_bool(bool val);
|
void add_bool(bool val);
|
||||||
|
void add_null();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void add_unquoted_str(const char* val);
|
void add_unquoted_str(const char* val);
|
||||||
|
@@ -815,10 +815,14 @@ void Explain_index_use::set_pseudo_key(MEM_ROOT *root, const char* key_name_arg)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
Given r_filtered% from join buffer condition and join condition, produce a
|
||||||
|
combined r_filtered% number. This is needed for tabular EXPLAIN output which
|
||||||
|
has only one cell for r_filtered value.
|
||||||
|
*/
|
||||||
|
|
||||||
double Explain_table_access::get_r_filtered()
|
double Explain_table_access::get_r_filtered()
|
||||||
{
|
{
|
||||||
//psergey-todo: modify this to produce separate filtered% for both parts of
|
|
||||||
//WHERE.
|
|
||||||
double r_filtered= tracker.get_filtered_after_where();
|
double r_filtered= tracker.get_filtered_after_where();
|
||||||
if (bka_type.is_using_jbuf())
|
if (bka_type.is_using_jbuf())
|
||||||
r_filtered *= jbuf_tracker.get_filtered_after_where();
|
r_filtered *= jbuf_tracker.get_filtered_after_where();
|
||||||
@@ -1156,15 +1160,27 @@ void Explain_table_access::print_explain_json(Explain_query *query,
|
|||||||
writer->end_array();
|
writer->end_array();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* r_loops (not present in tabular output) */
|
||||||
|
if (is_analyze)
|
||||||
|
{
|
||||||
|
writer->add_member("r_loops").add_ll(tracker.get_loops());
|
||||||
|
}
|
||||||
|
|
||||||
/* `rows` */
|
/* `rows` */
|
||||||
if (rows_set)
|
if (rows_set)
|
||||||
writer->add_member("rows").add_ll(rows);
|
writer->add_member("rows").add_ll(rows);
|
||||||
|
|
||||||
/* `r_rows` */
|
/* `r_rows` */
|
||||||
if (is_analyze && tracker.has_scans())
|
if (is_analyze)
|
||||||
{
|
{
|
||||||
ha_rows avg_rows= tracker.get_avg_rows();
|
writer->add_member("r_rows");
|
||||||
writer->add_member("r_rows").add_ll(avg_rows);
|
if (tracker.has_scans())
|
||||||
|
{
|
||||||
|
ha_rows avg_rows= tracker.get_avg_rows();
|
||||||
|
writer->add_ll(avg_rows);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
writer->add_null();
|
||||||
}
|
}
|
||||||
|
|
||||||
/* `filtered` */
|
/* `filtered` */
|
||||||
@@ -1173,7 +1189,13 @@ void Explain_table_access::print_explain_json(Explain_query *query,
|
|||||||
|
|
||||||
/* `r_filtered` */
|
/* `r_filtered` */
|
||||||
if (is_analyze)
|
if (is_analyze)
|
||||||
writer->add_member("r_filtered").add_double(get_r_filtered());
|
{
|
||||||
|
writer->add_member("r_filtered");
|
||||||
|
if (tracker.has_scans())
|
||||||
|
writer->add_double(tracker.get_filtered_after_where()*100.0);
|
||||||
|
else
|
||||||
|
writer->add_null();
|
||||||
|
}
|
||||||
|
|
||||||
for (int i=0; i < (int)extra_tags.elements(); i++)
|
for (int i=0; i < (int)extra_tags.elements(); i++)
|
||||||
{
|
{
|
||||||
@@ -1193,6 +1215,16 @@ void Explain_table_access::print_explain_json(Explain_query *query,
|
|||||||
writer->add_member("attached_condition");
|
writer->add_member("attached_condition");
|
||||||
write_item(writer, where_cond);
|
write_item(writer, where_cond);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (is_analyze)
|
||||||
|
{
|
||||||
|
//writer->add_member("r_loops").add_ll(jbuf_tracker.get_loops());
|
||||||
|
writer->add_member("r_filtered");
|
||||||
|
if (jbuf_tracker.has_scans())
|
||||||
|
writer->add_double(jbuf_tracker.get_filtered_after_where()*100.0);
|
||||||
|
else
|
||||||
|
writer->add_null();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (derived_select_number)
|
if (derived_select_number)
|
||||||
|
@@ -22,8 +22,13 @@ public:
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
A class for collecting read statistics.
|
||||||
|
|
||||||
|
The idea is that we run several scans. Each scans gets rows, and then filters
|
||||||
|
some of them out. We count scans, rows, and rows left after filtering.
|
||||||
|
*/
|
||||||
|
|
||||||
/* Data structures for ANALYZE */
|
|
||||||
class Table_access_tracker
|
class Table_access_tracker
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
@@ -38,6 +43,7 @@ public:
|
|||||||
ha_rows r_rows_after_where; /* Rows after applying attached part of WHERE */
|
ha_rows r_rows_after_where; /* Rows after applying attached part of WHERE */
|
||||||
|
|
||||||
bool has_scans() { return (r_scans != 0); }
|
bool has_scans() { return (r_scans != 0); }
|
||||||
|
ha_rows get_loops() { return r_scans; }
|
||||||
ha_rows get_avg_rows()
|
ha_rows get_avg_rows()
|
||||||
{
|
{
|
||||||
return r_scans ? (ha_rows)rint((double) r_rows / r_scans): 0;
|
return r_scans ? (ha_rows)rint((double) r_rows / r_scans): 0;
|
||||||
@@ -611,7 +617,9 @@ public:
|
|||||||
void print_explain_json(Explain_query *query, Json_writer *writer,
|
void print_explain_json(Explain_query *query, Json_writer *writer,
|
||||||
bool is_analyze);
|
bool is_analyze);
|
||||||
|
|
||||||
/* ANALYZE members*/
|
/* ANALYZE members */
|
||||||
|
|
||||||
|
/* Tracker for reading the table */
|
||||||
Table_access_tracker tracker;
|
Table_access_tracker tracker;
|
||||||
Table_access_tracker jbuf_tracker;
|
Table_access_tracker jbuf_tracker;
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user