mirror of
https://github.com/MariaDB/server.git
synced 2025-07-27 18:02:13 +03:00
MDEV-18741: Optimizer trace: multi-part key ranges are printed incorrectly
Changed the function append_range_all_keyparts to use sel_arg_range_seq_init / sel_arg_range_seq_next to produce ranges. Also adjusted to print format for the ranges, now the ranges are printed as: (keypart1_min, keypart2_min,..) OP (keypart1_name,keypart2_name, ..) OP (keypart1_max,keypart2_max, ..) Also added more tests for range and index merge access for optimizer trace
This commit is contained in:
@ -110,7 +110,7 @@ explain select * from t1 where a=1 or b=1 {
|
||||
"range_scan_alternatives": [
|
||||
{
|
||||
"index": "a",
|
||||
"ranges": ["1 <= a <= 1"],
|
||||
"ranges": ["(1) <= (a) <= (1)"],
|
||||
"rowid_ordered": true,
|
||||
"using_mrr": false,
|
||||
"index_only": true,
|
||||
@ -126,7 +126,7 @@ explain select * from t1 where a=1 or b=1 {
|
||||
"range_scan_alternatives": [
|
||||
{
|
||||
"index": "b",
|
||||
"ranges": ["1 <= b <= 1"],
|
||||
"ranges": ["(1) <= (b) <= (1)"],
|
||||
"rowid_ordered": true,
|
||||
"using_mrr": false,
|
||||
"index_only": true,
|
||||
@ -147,7 +147,7 @@ explain select * from t1 where a=1 or b=1 {
|
||||
"type": "range_scan",
|
||||
"index": "a",
|
||||
"rows": 1,
|
||||
"ranges": ["1 <= a <= 1"],
|
||||
"ranges": ["(1) <= (a) <= (1)"],
|
||||
"analyzing_roworder_intersect": {
|
||||
"cause": "too few roworder scans"
|
||||
}
|
||||
@ -156,7 +156,7 @@ explain select * from t1 where a=1 or b=1 {
|
||||
"type": "range_scan",
|
||||
"index": "b",
|
||||
"rows": 1,
|
||||
"ranges": ["1 <= b <= 1"],
|
||||
"ranges": ["(1) <= (b) <= (1)"],
|
||||
"analyzing_roworder_intersect": {
|
||||
"cause": "too few roworder scans"
|
||||
}
|
||||
@ -176,13 +176,13 @@ explain select * from t1 where a=1 or b=1 {
|
||||
"type": "range_scan",
|
||||
"index": "a",
|
||||
"rows": 1,
|
||||
"ranges": ["1 <= a <= 1"]
|
||||
"ranges": ["(1) <= (a) <= (1)"]
|
||||
},
|
||||
{
|
||||
"type": "range_scan",
|
||||
"index": "b",
|
||||
"rows": 1,
|
||||
"ranges": ["1 <= b <= 1"]
|
||||
"ranges": ["(1) <= (b) <= (1)"]
|
||||
}
|
||||
]
|
||||
},
|
||||
@ -243,3 +243,500 @@ explain select * from t1 where a=1 or b=1 {
|
||||
drop table t0,t1;
|
||||
set optimizer_trace="enabled=off";
|
||||
set @@optimizer_switch= @tmp_opt_switch;
|
||||
# More tests added index_merge access
|
||||
create table t1
|
||||
(
|
||||
/* Field names reflect value(rowid) distribution, st=STairs, swt= SaWTooth */
|
||||
st_a int not null default 0,
|
||||
swt1a int not null default 0,
|
||||
swt2a int not null default 0,
|
||||
st_b int not null default 0,
|
||||
swt1b int not null default 0,
|
||||
swt2b int not null default 0,
|
||||
/* fields/keys for row retrieval tests */
|
||||
key1 int,
|
||||
key2 int,
|
||||
key3 int,
|
||||
key4 int,
|
||||
/* make rows much bigger then keys */
|
||||
filler1 char (200),
|
||||
filler2 char (200),
|
||||
filler3 char (200),
|
||||
filler4 char (200),
|
||||
filler5 char (200),
|
||||
filler6 char (200),
|
||||
/* order of keys is important */
|
||||
key sta_swt12a(st_a,swt1a,swt2a),
|
||||
key sta_swt1a(st_a,swt1a),
|
||||
key sta_swt2a(st_a,swt2a),
|
||||
key sta_swt21a(st_a,swt2a,swt1a),
|
||||
key st_a(st_a),
|
||||
key stb_swt1a_2b(st_b,swt1b,swt2a),
|
||||
key stb_swt1b(st_b,swt1b),
|
||||
key st_b(st_b),
|
||||
key(key1),
|
||||
key(key2),
|
||||
key(key3),
|
||||
key(key4)
|
||||
) ;
|
||||
create table t0 as select * from t1;
|
||||
# Printing of many insert into t0 values (....) disabled.
|
||||
alter table t1 disable keys;
|
||||
# Printing of many insert into t1 select .... from t0 disabled.
|
||||
# Printing of many insert into t1 (...) values (....) disabled.
|
||||
alter table t1 enable keys;
|
||||
insert into t1 (key1, key2, key3, key4, filler1) values (100, 100, -1, -1, 'key1-key2');
|
||||
insert into t1 (key1, key2, key3, key4, filler1) values (-1, -1, 100, 100, 'key4-key3');
|
||||
set optimizer_trace='enabled=on';
|
||||
# 3-way ROR-intersection
|
||||
explain select key1,key2,key3 from t1 where key1=100 and key2=100 and key3=100;
|
||||
id select_type table type possible_keys key key_len ref rows Extra
|
||||
1 SIMPLE t1 index_merge key1,key2,key3 key1,key2,key3 5,5,5 NULL 2 Using intersect(key1,key2,key3); Using where; Using index
|
||||
select JSON_DETAILED(JSON_EXTRACT(trace, '$**.analyzing_range_alternatives')) from INFORMATION_SCHEMA.OPTIMIZER_TRACE;
|
||||
JSON_DETAILED(JSON_EXTRACT(trace, '$**.analyzing_range_alternatives'))
|
||||
[
|
||||
|
||||
{
|
||||
"range_scan_alternatives":
|
||||
[
|
||||
|
||||
{
|
||||
"index": "key1",
|
||||
"ranges":
|
||||
[
|
||||
"(100) <= (key1) <= (100)"
|
||||
],
|
||||
"rowid_ordered": true,
|
||||
"using_mrr": false,
|
||||
"index_only": false,
|
||||
"rows": 2243,
|
||||
"cost": 2862.1,
|
||||
"chosen": true
|
||||
},
|
||||
|
||||
{
|
||||
"index": "key2",
|
||||
"ranges":
|
||||
[
|
||||
"(100) <= (key2) <= (100)"
|
||||
],
|
||||
"rowid_ordered": true,
|
||||
"using_mrr": false,
|
||||
"index_only": false,
|
||||
"rows": 2243,
|
||||
"cost": 2862.1,
|
||||
"chosen": false,
|
||||
"cause": "cost"
|
||||
},
|
||||
|
||||
{
|
||||
"index": "key3",
|
||||
"ranges":
|
||||
[
|
||||
"(100) <= (key3) <= (100)"
|
||||
],
|
||||
"rowid_ordered": true,
|
||||
"using_mrr": false,
|
||||
"index_only": false,
|
||||
"rows": 2243,
|
||||
"cost": 2862.1,
|
||||
"chosen": false,
|
||||
"cause": "cost"
|
||||
}
|
||||
],
|
||||
"analyzing_roworder_intersect":
|
||||
{
|
||||
"intersecting_indexes":
|
||||
[
|
||||
|
||||
{
|
||||
"index": "key1",
|
||||
"index_scan_cost": 58.252,
|
||||
"cumulated_index_scan_cost": 58.252,
|
||||
"disk_sweep_cost": 1923.1,
|
||||
"cumulative_total_cost": 1981.4,
|
||||
"usable": true,
|
||||
"matching_rows_now": 2243,
|
||||
"intersect_covering_with_this_index": false,
|
||||
"chosen": true
|
||||
},
|
||||
|
||||
{
|
||||
"index": "key2",
|
||||
"index_scan_cost": 58.252,
|
||||
"cumulated_index_scan_cost": 116.5,
|
||||
"disk_sweep_cost": 84.518,
|
||||
"cumulative_total_cost": 201.02,
|
||||
"usable": true,
|
||||
"matching_rows_now": 77.636,
|
||||
"intersect_covering_with_this_index": false,
|
||||
"chosen": true
|
||||
},
|
||||
|
||||
{
|
||||
"index": "key3",
|
||||
"index_scan_cost": 58.252,
|
||||
"cumulated_index_scan_cost": 174.76,
|
||||
"disk_sweep_cost": 0,
|
||||
"cumulative_total_cost": 174.76,
|
||||
"usable": true,
|
||||
"matching_rows_now": 2.6872,
|
||||
"intersect_covering_with_this_index": true,
|
||||
"chosen": true
|
||||
}
|
||||
],
|
||||
"clustered_pk":
|
||||
{
|
||||
"clustered_pk_added_to_intersect": false,
|
||||
"cause": "no clustered pk index"
|
||||
},
|
||||
"rows": 2,
|
||||
"cost": 174.76,
|
||||
"covering": true,
|
||||
"chosen": true
|
||||
},
|
||||
"analyzing_index_merge_union":
|
||||
[
|
||||
]
|
||||
}
|
||||
]
|
||||
select JSON_DETAILED(JSON_EXTRACT(trace, '$**.chosen_range_access_summary')) from INFORMATION_SCHEMA.OPTIMIZER_TRACE;
|
||||
JSON_DETAILED(JSON_EXTRACT(trace, '$**.chosen_range_access_summary'))
|
||||
[
|
||||
|
||||
{
|
||||
"range_access_plan":
|
||||
{
|
||||
"type": "index_roworder_intersect",
|
||||
"rows": 2,
|
||||
"cost": 174.76,
|
||||
"covering": true,
|
||||
"clustered_pk_scan": false,
|
||||
"intersect_of":
|
||||
[
|
||||
|
||||
{
|
||||
"type": "range_scan",
|
||||
"index": "key1",
|
||||
"rows": 2243,
|
||||
"ranges":
|
||||
[
|
||||
"(100) <= (key1) <= (100)"
|
||||
]
|
||||
},
|
||||
|
||||
{
|
||||
"type": "range_scan",
|
||||
"index": "key2",
|
||||
"rows": 2243,
|
||||
"ranges":
|
||||
[
|
||||
"(100) <= (key2) <= (100)"
|
||||
]
|
||||
},
|
||||
|
||||
{
|
||||
"type": "range_scan",
|
||||
"index": "key3",
|
||||
"rows": 2243,
|
||||
"ranges":
|
||||
[
|
||||
"(100) <= (key3) <= (100)"
|
||||
]
|
||||
}
|
||||
]
|
||||
},
|
||||
"rows_for_plan": 2,
|
||||
"cost_for_plan": 174.76,
|
||||
"chosen": true
|
||||
}
|
||||
]
|
||||
# ROR-union(ROR-intersection, ROR-range)
|
||||
explain select key1,key2,key3,key4 from t1 where key1=100 and key2=100 or key3=100 and key4=100;
|
||||
id select_type table type possible_keys key key_len ref rows Extra
|
||||
1 SIMPLE t1 index_merge key1,key2,key3,key4 key1,key2,key3,key4 5,5,5,5 NULL 154 Using union(intersect(key1,key2),intersect(key3,key4)); Using where
|
||||
select JSON_DETAILED(JSON_EXTRACT(trace, '$**.analyzing_range_alternatives')) from INFORMATION_SCHEMA.OPTIMIZER_TRACE;
|
||||
JSON_DETAILED(JSON_EXTRACT(trace, '$**.analyzing_range_alternatives'))
|
||||
[
|
||||
|
||||
{
|
||||
"range_scan_alternatives":
|
||||
[
|
||||
],
|
||||
"analyzing_roworder_intersect":
|
||||
{
|
||||
"cause": "too few roworder scans"
|
||||
},
|
||||
"analyzing_index_merge_union":
|
||||
[
|
||||
|
||||
{
|
||||
"indexes_to_merge":
|
||||
[
|
||||
|
||||
{
|
||||
"range_scan_alternatives":
|
||||
[
|
||||
|
||||
{
|
||||
"index": "key1",
|
||||
"ranges":
|
||||
[
|
||||
"(100) <= (key1) <= (100)"
|
||||
],
|
||||
"rowid_ordered": true,
|
||||
"using_mrr": false,
|
||||
"index_only": true,
|
||||
"rows": 2243,
|
||||
"cost": 170.53,
|
||||
"chosen": true
|
||||
},
|
||||
|
||||
{
|
||||
"index": "key2",
|
||||
"ranges":
|
||||
[
|
||||
"(100) <= (key2) <= (100)"
|
||||
],
|
||||
"rowid_ordered": true,
|
||||
"using_mrr": false,
|
||||
"index_only": true,
|
||||
"rows": 2243,
|
||||
"cost": 170.53,
|
||||
"chosen": false,
|
||||
"cause": "cost"
|
||||
}
|
||||
],
|
||||
"index_to_merge": "key1",
|
||||
"cumulated_cost": 170.53
|
||||
},
|
||||
|
||||
{
|
||||
"range_scan_alternatives":
|
||||
[
|
||||
|
||||
{
|
||||
"index": "key3",
|
||||
"ranges":
|
||||
[
|
||||
"(100) <= (key3) <= (100)"
|
||||
],
|
||||
"rowid_ordered": true,
|
||||
"using_mrr": false,
|
||||
"index_only": true,
|
||||
"rows": 2243,
|
||||
"cost": 170.53,
|
||||
"chosen": true
|
||||
},
|
||||
|
||||
{
|
||||
"index": "key4",
|
||||
"ranges":
|
||||
[
|
||||
"(100) <= (key4) <= (100)"
|
||||
],
|
||||
"rowid_ordered": true,
|
||||
"using_mrr": false,
|
||||
"index_only": true,
|
||||
"rows": 2243,
|
||||
"cost": 170.53,
|
||||
"chosen": false,
|
||||
"cause": "cost"
|
||||
}
|
||||
],
|
||||
"index_to_merge": "key3",
|
||||
"cumulated_cost": 341.05
|
||||
}
|
||||
],
|
||||
"cost_of_reading_ranges": 341.05,
|
||||
"use_roworder_union": true,
|
||||
"cause": "always cheaper than non roworder retrieval",
|
||||
"analyzing_roworder_scans":
|
||||
[
|
||||
|
||||
{
|
||||
"type": "range_scan",
|
||||
"index": "key1",
|
||||
"rows": 2243,
|
||||
"ranges":
|
||||
[
|
||||
"(100) <= (key1) <= (100)"
|
||||
],
|
||||
"analyzing_roworder_intersect":
|
||||
{
|
||||
"intersecting_indexes":
|
||||
[
|
||||
|
||||
{
|
||||
"index": "key1",
|
||||
"index_scan_cost": 58.252,
|
||||
"cumulated_index_scan_cost": 58.252,
|
||||
"disk_sweep_cost": 1923.1,
|
||||
"cumulative_total_cost": 1981.4,
|
||||
"usable": true,
|
||||
"matching_rows_now": 2243,
|
||||
"intersect_covering_with_this_index": false,
|
||||
"chosen": true
|
||||
},
|
||||
|
||||
{
|
||||
"index": "key2",
|
||||
"index_scan_cost": 58.252,
|
||||
"cumulated_index_scan_cost": 116.5,
|
||||
"disk_sweep_cost": 84.518,
|
||||
"cumulative_total_cost": 201.02,
|
||||
"usable": true,
|
||||
"matching_rows_now": 77.636,
|
||||
"intersect_covering_with_this_index": false,
|
||||
"chosen": true
|
||||
}
|
||||
],
|
||||
"clustered_pk":
|
||||
{
|
||||
"clustered_pk_added_to_intersect": false,
|
||||
"cause": "no clustered pk index"
|
||||
},
|
||||
"rows": 77,
|
||||
"cost": 201.02,
|
||||
"covering": false,
|
||||
"chosen": true
|
||||
}
|
||||
},
|
||||
|
||||
{
|
||||
"type": "range_scan",
|
||||
"index": "key3",
|
||||
"rows": 2243,
|
||||
"ranges":
|
||||
[
|
||||
"(100) <= (key3) <= (100)"
|
||||
],
|
||||
"analyzing_roworder_intersect":
|
||||
{
|
||||
"intersecting_indexes":
|
||||
[
|
||||
|
||||
{
|
||||
"index": "key3",
|
||||
"index_scan_cost": 58.252,
|
||||
"cumulated_index_scan_cost": 58.252,
|
||||
"disk_sweep_cost": 1923.1,
|
||||
"cumulative_total_cost": 1981.4,
|
||||
"usable": true,
|
||||
"matching_rows_now": 2243,
|
||||
"intersect_covering_with_this_index": false,
|
||||
"chosen": true
|
||||
},
|
||||
|
||||
{
|
||||
"index": "key4",
|
||||
"index_scan_cost": 58.252,
|
||||
"cumulated_index_scan_cost": 116.5,
|
||||
"disk_sweep_cost": 84.518,
|
||||
"cumulative_total_cost": 201.02,
|
||||
"usable": true,
|
||||
"matching_rows_now": 77.636,
|
||||
"intersect_covering_with_this_index": false,
|
||||
"chosen": true
|
||||
}
|
||||
],
|
||||
"clustered_pk":
|
||||
{
|
||||
"clustered_pk_added_to_intersect": false,
|
||||
"cause": "no clustered pk index"
|
||||
},
|
||||
"rows": 77,
|
||||
"cost": 201.02,
|
||||
"covering": false,
|
||||
"chosen": true
|
||||
}
|
||||
}
|
||||
],
|
||||
"index_roworder_union_cost": 386.73,
|
||||
"members": 2,
|
||||
"chosen": true
|
||||
}
|
||||
]
|
||||
}
|
||||
]
|
||||
select JSON_DETAILED(JSON_EXTRACT(trace, '$**.chosen_range_access_summary')) from INFORMATION_SCHEMA.OPTIMIZER_TRACE;
|
||||
JSON_DETAILED(JSON_EXTRACT(trace, '$**.chosen_range_access_summary'))
|
||||
[
|
||||
|
||||
{
|
||||
"range_access_plan":
|
||||
{
|
||||
"type": "index_roworder_union",
|
||||
"union_of":
|
||||
[
|
||||
|
||||
{
|
||||
"type": "index_roworder_intersect",
|
||||
"rows": 77,
|
||||
"cost": 201.02,
|
||||
"covering": false,
|
||||
"clustered_pk_scan": false,
|
||||
"intersect_of":
|
||||
[
|
||||
|
||||
{
|
||||
"type": "range_scan",
|
||||
"index": "key1",
|
||||
"rows": 2243,
|
||||
"ranges":
|
||||
[
|
||||
"(100) <= (key1) <= (100)"
|
||||
]
|
||||
},
|
||||
|
||||
{
|
||||
"type": "range_scan",
|
||||
"index": "key2",
|
||||
"rows": 2243,
|
||||
"ranges":
|
||||
[
|
||||
"(100) <= (key2) <= (100)"
|
||||
]
|
||||
}
|
||||
]
|
||||
},
|
||||
|
||||
{
|
||||
"type": "index_roworder_intersect",
|
||||
"rows": 77,
|
||||
"cost": 201.02,
|
||||
"covering": false,
|
||||
"clustered_pk_scan": false,
|
||||
"intersect_of":
|
||||
[
|
||||
|
||||
{
|
||||
"type": "range_scan",
|
||||
"index": "key3",
|
||||
"rows": 2243,
|
||||
"ranges":
|
||||
[
|
||||
"(100) <= (key3) <= (100)"
|
||||
]
|
||||
},
|
||||
|
||||
{
|
||||
"type": "range_scan",
|
||||
"index": "key4",
|
||||
"rows": 2243,
|
||||
"ranges":
|
||||
[
|
||||
"(100) <= (key4) <= (100)"
|
||||
]
|
||||
}
|
||||
]
|
||||
}
|
||||
]
|
||||
},
|
||||
"rows_for_plan": 154,
|
||||
"cost_for_plan": 386.73,
|
||||
"chosen": true
|
||||
}
|
||||
]
|
||||
drop table t0,t1;
|
||||
set optimizer_trace="enabled=off";
|
||||
|
Reference in New Issue
Block a user