have close costs. And indeed, the execution times for the two join orders (with ORDER BY ... LIMIT removed) are close.
TMP-TABLE-COST is a small fraction of join cost and doesn't make a difference (this seems to match the reality, too).
What does make a difference is short-cutting of execution due to LIMIT, but this is not taken into account.
Doing it in general case requires everything from MDEV-8306.
This MDEV is to have a fix for the join optimizer which adjusts the join order cost for cases with very small limit.
The new behavior is to be controlled by a setting.
If one runs OLTP-mostly workloads, enabling new behavior is an obvious choice.
Implementation
First, identify sort_by_table - the table that allows to use LIMIT short-cutting.
(Due to multiple equalities there can be multiple but we can assume it's just one table)
Do adjustment of cost/cardinality only for full join orders
Let's assume we've built a complete join order (ignoring ORDER BY ... LIMIT).
Having done that, we can compare join_output_size with the LIMIT number and see which "fraction" of join output we need and adjust the cost accordingly.
(An alternative to this: try to account for ORDER BY ...LIMIT earlier, while building join prefixes. We don't try to do this).
Adjusting join order cost and cardinality
Do it as follows:
1: If the join output is already produced in the required order, we can assume that $FRACT of join output will take $FRACT of the cost to produce. (Note: this is incorrect when using e.g. semi-join materialization or materialized derived tables. We ignore this fact for now).
Otherwise, we have two options:
2A: Change the access method for sort_by_table so that we do get rows in the desired order. After that, we can take a fraction of the cost like described above.
2B. Pass sort_by_table to filesort. We will need to spend the entire cost of reading sort_by_table but then we will be able to short-cut joining with subsequent tables.
Interplay with join optimizer pruning
So, we will adjust the costs only for complete join orders. The adjustment will reduce the cost and output-records.
Join pruning compares costs of prefix with full join order costs. Comparing un-adjusted costs with adjusted can produce wrong results.
we can prune un-adjusted prefix that will be adjusted
Hooking into the join optimizer - variant 1
First, run the join optimization as usual. This produces QUERY_PLAN1 and a tight upper bound of how many records the join would produce. (If it's smaller than LIMIT, short-cutting will probably not be beneficial).
Save join->best_read and join->best_positions.
Then, start the optimization again:
set join->best_read= DBL_MAX, Run the join optimization with first table=sort_by_table.
This produces QUERY_PLAN2 query plan that can take advantage of short-cutting.
Normally it's more expensive than QUERY_PLAN1.
Now, account for LIMIT short-cutting as specified in Adjusting join order's cost and cardinality section. This produces cost of QUERY_PLAN2-WITH-SHORTCUTTING. We can compare it with the cost of QUERY_PLAN1.
Hooking into the join optimizer - variant 2
Make the join optimizer first start with sort_table as the first table.
Once we have constructed a full join order, perform #rows/cost adjustments for LIMIT.
We can set the adjusted cost as join->best_read.
Then, proceed to run the join optimizer as usual.
Account for risks
QUERY_PLAN2-WITH-SHORTCUTTING is inherently more "risky". For example, if the join has a condition with selectivity=0.5 that the optimizer is not aware of, the actual cost of producing #LIMIT rows will be 1/0.5=2 times higher.
To account for this, we will add a multiplier, e.g. use QUERY_PLAN2-WITH-SHORTCUTTING only if promises a 100x speedup over QUERY_PLAN1.
What about cost model and related changes in 11.0?
test_if_skip_sort_order() function itself is almost unchanged.
it calls test_if_cheaper_ordering(). There, there are some changes, some code seems to be moved into the get_range_limit_read_cost() calls.
Cost of sorting is added, see cost_of_filesort().
In get_range_limit_read_cost, there are changes.
Sergei Petrunia
added a comment - What about cost model and related changes in 11.0?
test_if_skip_sort_order() function itself is almost unchanged.
it calls test_if_cheaper_ordering() . There, there are some changes, some code seems to be moved into the get_range_limit_read_cost() calls.
Cost of sorting is added, see cost_of_filesort().
In get_range_limit_read_cost , there are changes.
Can we just build a join order and then call test_if_skip_sort_order()?
No:
even with no_changes=true parameter, test_if_skip_sort_order(... no_changes=true...) will do this:
Access tab->ref.*, tab->type. These are only usable after join optimization has finished and get_best_combination() was called.
Change tab->quick to hold different quick select.
What about other functions used from test_if_skip_sort_order?
get_range_limit_read_cost(const JOIN_TAB *tab, ... ) CAN be used at join optimization stage. It only uses tab->worst_seeks.
test_if_cheaper_ordering(const JOIN_TAB *tab, ...) CANNOT be used at join optimization stage: it uses tab->type and tab->ref which are not yet set.
It also does this: "uint tablenr= (uint)(tab - join->join_tab);" - this is valid only after get_best_combination re-arranged to JOIN_TABs to follow the picked join order.
Sergei Petrunia
added a comment - - edited Can we just build a join order and then call test_if_skip_sort_order()?
No:
even with no_changes=true parameter, test_if_skip_sort_order(... no_changes=true...) will do this:
Access tab->ref.*, tab->type. These are only usable after join optimization has finished and get_best_combination() was called.
Change tab->quick to hold different quick select.
What about other functions used from test_if_skip_sort_order?
get_range_limit_read_cost(const JOIN_TAB *tab, ... ) CAN be used at join optimization stage. It only uses tab->worst_seeks.
test_if_cheaper_ordering(const JOIN_TAB *tab, ...) CANNOT be used at join optimization stage: it uses tab->type and tab->ref which are not yet set.
It also does this: "uint tablenr= (uint)(tab - join->join_tab);" - this is valid only after get_best_combination re-arranged to JOIN_TABs to follow the picked join order.
A question: what is the point of looking for a suitable key in test_if_skip_sort_order() here:
if ((new_ref_key= test_if_subkey(order, table, ref_key, ref_key_parts,
&usable_keys)) < MAX_KEY)
if then in test_if_cheaper_ordering() we try all keys anyway?
Sergei Petrunia
added a comment - A question: what is the point of looking for a suitable key in test_if_skip_sort_order() here:
if ((new_ref_key= test_if_subkey(order, table, ref_key, ref_key_parts,
&usable_keys)) < MAX_KEY)
if then in test_if_cheaper_ordering() we try all keys anyway?
Note: What about commit 515b9ad05 for MDEV-28073 which introduces sorting of tables and get_costs_for_tables()?
The sorting done after get_costs_for_tables() call may put some other table ahead of sort_table and the requirement that "join orders starting with sort_table are considered before the any other join orders" will be broken.
Limited lookahead
What if greedy optimization uses a limited lookahead?
Produce a limited prefix starting with sort_table;
Extend it until we have build a full join order;
Adjust the cost with join_limit_shortcut_adjust_cost();
Then we need to restart the process. (this cannot be done from inside best_extension_by_limited_search).
We need to
Build a limited prefix from scratch; // with no limitation on the first table
Extend it until we have build a full join order;
The easiest way to achieve this is to call choose_plan twice... Everything else seems to be too error-prone.
Sergei Petrunia
added a comment - - edited On the suggestion to not make two calls to greedy_search():
Exhaustive search
This works well when choose_plan is doing exhaustive search:
In choose_plan():
Put the sort_by_table first into join->best_ref;
the rest of join->best_ref() is sorted according to jtab_sort_func
Then, exhaustive search will enumerate prefixes starting from sort_table first.
Add a hook at the end of the main loop in best_extension_by_limited_search():
if (idx == join->const_tables && // we're changing first table
join->positions[idx] == sort_by_table) // currently it's sort_by_table
{
// We have finished considering join prefixes starting with sort_table
DBUG_ASSERT(join->best_positions[idx].table == sort_by_table);
join_limit_shortcut_adjust_cost();
}
Note: What about commit 515b9ad05 for MDEV-28073 which introduces sorting of tables and get_costs_for_tables()?
The sorting done after get_costs_for_tables() call may put some other table ahead of sort_table and the requirement that "join orders starting with sort_table are considered before the any other join orders" will be broken.
Limited lookahead
What if greedy optimization uses a limited lookahead?
Produce a limited prefix starting with sort_table;
Extend it until we have build a full join order;
Adjust the cost with join_limit_shortcut_adjust_cost();
Then we need to restart the process. (this cannot be done from inside best_extension_by_limited_search).
We need to
Build a limited prefix from scratch; // with no limitation on the first table
Extend it until we have build a full join order;
The easiest way to achieve this is to call choose_plan twice... Everything else seems to be too error-prone.
Don't allow limited lookahead when the limit optimization in enabled and we can use the optimization.
Same as above, but do not do limit lookhead for the order-by-first table (don't know if this is possible)
Use your old code to handle this case. This code still can benefit is having the order-by table first.
Michael Widenius
added a comment - A few ways to solve this:
Don't allow limited lookahead when the limit optimization in enabled and we can use the optimization.
Same as above, but do not do limit lookhead for the order-by-first table (don't know if this is possible)
Use your old code to handle this case. This code still can benefit is having the order-by table first.
join->best_read is the cost of entire join order (adjusted for LIMIT-shortcut)
current_read_time is the cost of join prefix.
It is clear that these two are not comparable.
Sergei Petrunia
added a comment - There is a problem with the current patch and greedy (aka limited-lookahead ) optimization.
The first greedy_search() call results in a complete plan in join->best_positions and its cost in join->best_read .
But then the second greedy_search() call is made.
We build a join prefix of seaarch_depth size:
greedy_search()
best_extension_by_limited_search(...search_depth=N...)
best_extension_by_limited_search(...search_depth=N-1...)
...
best_extension_by_limited_search(...search_depth=1...)
Here, best_extension_by_limited_search() will do this:
if (current_read_time < join->best_read)
{
memcpy ((uchar*) join->best_positions, (uchar*) join->positions,
sizeof (POSITION) * (idx + 1));
join->join_record_count= partial_join_cardinality;
join->best_read= current_read_time - 0.001;
}
Here,
join->best_read is the cost of entire join order (adjusted for LIMIT-shortcut)
current_read_time is the cost of join prefix.
It is clear that these two are not comparable.
MDEV-34720: Poor plan choice for large JOIN with ORDER BY and small LIMIT
(Variant 2b: call greedy_search() twice, correct handling for limited
search_depth)
Sergei Petrunia
added a comment - Fixed the above
commit 4c00931dafbb87edeac6afe44b2f6eeb3c6afc61 (HEAD -> bb-10.6-mdev34720-v2, origin/bb-10.6-mdev34720-v2)
Author: Sergei Petrunia <sergey@mariadb.com>
Date: Sun Aug 18 20:10:19 2024 +0300
MDEV-34720: Poor plan choice for large JOIN with ORDER BY and small LIMIT
(Variant 2b: call greedy_search() twice, correct handling for limited
search_depth)
Sergei Petrunia
added a comment - - edited More details about test_if_cheaper_ordering using rec_per_key and not actual_rec_per_key()
Test case from subselect_innodb shows:
explain select
( SELECT
concat(id, '-' , key1, '-' , col1)
FROM t2
WHERE t2.key1 = t1.a
ORDER BY t2.key2 ASC LIMIT 1)
from
t1;
In best_access_path(s=t2)
/* Check if we have statistic about the distribution */
if ((records= keyinfo->actual_rec_per_key(max_key_part-1)))
records=100.
Then, in test_if_cheaper_ordering(tab=t2) rec_per_key is used:
/*
Calculate the selectivity of the ref_key for REF_ACCESS. For
RANGE_ACCESS we use table->opt_range_condition_rows.
*/
if (ref_key >= 0 && ref_key != MAX_KEY && tab->type == JT_REF)
{
...
else
{
const KEY *ref_keyinfo= table->key_info + ref_key;
refkey_rows_estimate= ref_keyinfo->rec_per_key[tab-> ref .key_parts - 1];
}
set_if_bigger(refkey_rows_estimate, 1);
This shows refkey_rows_estimate=50
Trivial fix: Make test_if_cheaper_ordering() use actual_rec_per_key()
Discovered this while working on MDEV-34720: test_if_cheaper_ordering()
uses rec_per_key, while the original estimate for the access method
is produced in best_access_path() by using actual_rec_per_key().
Make test_if_cheaper_ordering() also use actual_rec_per_key().
Also make several getter function "const" to make this compile.
Also adjusted the testcase to handle this (the change backported from
11.0)
(This is not necessary for the patch but will be useful when getting the patch to the main tree)
Sergei Petrunia
added a comment - - edited For the above, pushed
commit 9020baf126c7d4068b40c57bc2d7dcb747b4b341 (HEAD -> 10.6)
Author: Sergei Petrunia <sergey@mariadb.com>
Date: Sat Aug 24 19:01:06 2024 +0300
Trivial fix: Make test_if_cheaper_ordering() use actual_rec_per_key()
Discovered this while working on MDEV-34720: test_if_cheaper_ordering()
uses rec_per_key, while the original estimate for the access method
is produced in best_access_path() by using actual_rec_per_key().
Make test_if_cheaper_ordering() also use actual_rec_per_key().
Also make several getter function "const" to make this compile.
Also adjusted the testcase to handle this (the change backported from
11.0)
(This is not necessary for the patch but will be useful when getting the patch to the main tree)
MDEV-34720: Poor plan choice for large JOIN with ORDER BY and small LIMIT
(Variant 2b: call greedy_search() twice, correct handling for limited
search_depth)
Modify the join optimizer to specifically try to produce join orders that
can short-cut their execution for ORDER BY..LIMIT clause.
The optimization is controlled by @@optimizer_join_limit_pref_ratio.
Default value 0 means don't construct short-cutting join orders.
Other value means construct short-cutting join order, and prefer it only
if it promises speedup of more than #value times.
In Optimizer Trace, look for these names:
* join_limit_shortcut_is_applicable
* join_limit_shortcut_plan_search
* join_limit_shortcut_choice
Sergei Petrunia
added a comment - The patch for the issue itself, latest revision:
commit 4c00931dafbb87edeac6afe44b2f6eeb3c6afc61 (HEAD -> bb-10.6-mdev34720-v2, origin/bb-10.6-mdev34720-v2)
Author: Sergei Petrunia <sergey@mariadb.com>
Date: Sun Aug 18 20:10:19 2024 +0300
MDEV-34720: Poor plan choice for large JOIN with ORDER BY and small LIMIT
(Variant 2b: call greedy_search() twice, correct handling for limited
search_depth)
Modify the join optimizer to specifically try to produce join orders that
can short-cut their execution for ORDER BY..LIMIT clause.
The optimization is controlled by @@optimizer_join_limit_pref_ratio.
Default value 0 means don't construct short-cutting join orders.
Other value means construct short-cutting join order, and prefer it only
if it promises speedup of more than #value times.
In Optimizer Trace, look for these names:
* join_limit_shortcut_is_applicable
* join_limit_shortcut_plan_search
* join_limit_shortcut_choice
Sergei Petrunia
added a comment - Documentation: https://mariadb.com/kb/en/optimizer_join_limit_pref_ratio-optimization/#providing-guidelines-to-the-optimizer
psergei, was a 10.11 version of this implemented, tested and reviewed? There are conflicts when attempting to merge this to 10.11.
Marko Mäkelä
added a comment - psergei , was a 10.11 version of this implemented, tested and reviewed? There are conflicts when attempting to merge this to 10.11.
Yuchen Pei
added a comment - - edited marko and psergei : I also noticed some non-trivial conflict when trying to merge 10.6->10.11 today, mainly with the following commits:
commit b3c74bdc1f09b2b8babd7f2bd9d52df2749ddcc3
Author: Monty <monty@mariadb.org>
Date: Tue May 31 17:36:32 2022 +0300
Improve pruning in greedy_search by sorting tables during search
MDEV-28073 Slow query performance in MariaDB when using many tables
...
commit 8c2faad576d6a77314e92755a389de2c41e21242 (github/bb-10.10-optimizer-features)
Author: Sergei Petrunia <sergey@mariadb.com>
Date: Tue Jul 19 14:13:17 2022 +0300
MDEV-28929: Plan selection takes forever with MDEV-28852 ...
Part #2: Extend heuristic pruning to use multiple tables as the
"Model tables".
...
The nontrivial conflict is this one in sql_select.cc:
@@@ -10738,98 -10853,31 +11096,109 @@@ best_extension_by_limited_search(JOI
DBUG_EXECUTE("opt", print_plan(join, idx, record_count, read_time, read_time,
"part_plan"););
# [... 32 lines elided]
<<<<<<< A: HEAD
# [... 57 lines elided]
for (SORT_POSITION *pos= sort ; pos < sort_end ; pos++)
{
s= *pos->join_tab;
if (!(found_eq_ref_tables & s->table->map) &&
!check_interleaving_with_nj(s))
||||||| Ancestor
if ((allowed_tables & real_table_bit) &&
!(remaining_tables & s->dependent) &&
!check_interleaving_with_nj(s))
=======
if ((allowed_tables & real_table_bit) &&
!(remaining_tables & s->dependent) &&
!check_interleaving_with_nj(s) &&
join_limit_shortcut_allows_table(join, idx, s))
>>>>>>> B: Incoming
{
+ table_map real_table_bit= s->table->map;
double current_record_count, current_read_time;
double partial_join_cardinality;
- POSITION *position= join->positions + idx;
- POSITION loose_scan_pos;
+ POSITION *position= join->positions + idx, *loose_scan_pos;
Json_writer_object trace_one_table(thd);
Can you advise how to resolve it, psergei ?
I just pushed a merge of this to 10.11 using this for reference. There will be the following conflicts left for a merge to 11.2:
both modified: mysql-test/main/mysqld--help.result
both modified: mysql-test/suite/sys_vars/r/sysvars_server_embedded.result
both modified: mysql-test/suite/sys_vars/r/sysvars_server_notembedded.result
both modified: sql/sql_select.cc
I didn’t check if there would be further conflicts for a merge to 11.4.
Marko Mäkelä
added a comment - I just pushed a merge of this to 10.11 using this for reference. There will be the following conflicts left for a merge to 11.2:
both modified: mysql-test/main/mysqld--help.result
both modified: mysql-test/suite/sys_vars/r/sysvars_server_embedded.result
both modified: mysql-test/suite/sys_vars/r/sysvars_server_notembedded.result
both modified: sql/sql_select.cc
I didn’t check if there would be further conflicts for a merge to 11.4.
ycp, I see that you must have been trying to merge 10.6→10.11 in order to resolve conflicts that your recently pushed changes introduced. You should be able to continue with that now.
Marko Mäkelä
added a comment - ycp , I see that you must have been trying to merge 10.6→10.11 in order to resolve conflicts that your recently pushed changes introduced. You should be able to continue with that now.
People
Sergei Petrunia
Sergei Petrunia
Votes:
1Vote for this issue
Watchers:
10Start watching this issue
Dates
Created:
Updated:
Resolved:
Git Integration
Error rendering 'com.xiplink.jira.git.jira_git_plugin:git-issue-webpanel'. Please contact your Jira administrators.
{"report":{"fcp":1588.3000001907349,"ttfb":601.3000001907349,"pageVisibility":"visible","entityId":130284,"key":"jira.project.issue.view-issue","isInitial":true,"threshold":1000,"elementTimings":{},"userDeviceMemory":8,"userDeviceProcessors":64,"apdex":0.5,"journeyId":"923fbc7c-afa1-461e-be01-e0a327c7fbd9","navigationType":0,"readyForUser":1689,"redirectCount":0,"resourceLoadedEnd":1968,"resourceLoadedStart":607.6999998092651,"resourceTiming":[{"duration":453.70000076293945,"initiatorType":"link","name":"https://jira.mariadb.org/s/2c21342762a6a02add1c328bed317ffd-CDN/lu2cib/820016/12ta74/0a8bac35585be7fc6c9cc5a0464cd4cf/_/download/contextbatch/css/_super/batch.css","startTime":607.6999998092651,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":607.6999998092651,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":1061.4000005722046,"responseStart":0,"secureConnectionStart":0},{"duration":453.5999994277954,"initiatorType":"link","name":"https://jira.mariadb.org/s/7ebd35e77e471bc30ff0eba799ebc151-CDN/lu2cib/820016/12ta74/494e4c556ecbb29f90a3d3b4f09cb99c/_/download/contextbatch/css/jira.browse.project,project.issue.navigator,jira.view.issue,jira.general,jira.global,atl.general,-_super/batch.css?agile_global_admin_condition=true&jag=true&jira.create.linked.issue=true&slack-enabled=true&whisper-enabled=true","startTime":608.1000003814697,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":608.1000003814697,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":1061.6999998092651,"responseStart":0,"secureConnectionStart":0},{"duration":506.9000005722046,"initiatorType":"script","name":"https://jira.mariadb.org/s/0917945aaa57108d00c5076fea35e069-CDN/lu2cib/820016/12ta74/0a8bac35585be7fc6c9cc5a0464cd4cf/_/download/contextbatch/js/_super/batch.js?locale=en","startTime":608.1999998092651,"connectEnd":608.1999998092651,"connectStart":608.1999998092651,"domainLookupEnd":608.1999998092651,"domainLookupStart":608.1999998092651,"fetchStart":608.1999998092651,"redirectEnd":0,"redirectStart":0,"requestStart":608.1999998092651,"responseEnd":1115.1000003814697,"responseStart":1115,"secureConnectionStart":608.1999998092651},{"duration":556.8999996185303,"initiatorType":"script","name":"https://jira.mariadb.org/s/2d8175ec2fa4c816e8023260bd8c1786-CDN/lu2cib/820016/12ta74/494e4c556ecbb29f90a3d3b4f09cb99c/_/download/contextbatch/js/jira.browse.project,project.issue.navigator,jira.view.issue,jira.general,jira.global,atl.general,-_super/batch.js?agile_global_admin_condition=true&jag=true&jira.create.linked.issue=true&locale=en&slack-enabled=true&whisper-enabled=true","startTime":608.4000005722046,"connectEnd":608.4000005722046,"connectStart":608.4000005722046,"domainLookupEnd":608.4000005722046,"domainLookupStart":608.4000005722046,"fetchStart":608.4000005722046,"redirectEnd":0,"redirectStart":0,"requestStart":608.4000005722046,"responseEnd":1165.3000001907349,"responseStart":1165.3000001907349,"secureConnectionStart":608.4000005722046},{"duration":560.8999996185303,"initiatorType":"script","name":"https://jira.mariadb.org/s/a9324d6758d385eb45c462685ad88f1d-CDN/lu2cib/820016/12ta74/c92c0caa9a024ae85b0ebdbed7fb4bd7/_/download/contextbatch/js/atl.global,-_super/batch.js?locale=en","startTime":608.8000001907349,"connectEnd":608.8000001907349,"connectStart":608.8000001907349,"domainLookupEnd":608.8000001907349,"domainLookupStart":608.8000001907349,"fetchStart":608.8000001907349,"redirectEnd":0,"redirectStart":0,"requestStart":608.8000001907349,"responseEnd":1169.6999998092651,"responseStart":1169.6999998092651,"secureConnectionStart":608.8000001907349},{"duration":561.2999992370605,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/1.0/_/download/batch/jira.webresources:calendar-en/jira.webresources:calendar-en.js","startTime":608.9000005722046,"connectEnd":608.9000005722046,"connectStart":608.9000005722046,"domainLookupEnd":608.9000005722046,"domainLookupStart":608.9000005722046,"fetchStart":608.9000005722046,"redirectEnd":0,"redirectStart":0,"requestStart":608.9000005722046,"responseEnd":1170.1999998092651,"responseStart":1170.1999998092651,"secureConnectionStart":608.9000005722046},{"duration":561.5,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/1.0/_/download/batch/jira.webresources:calendar-localisation-moment/jira.webresources:calendar-localisation-moment.js","startTime":609,"connectEnd":609,"connectStart":609,"domainLookupEnd":609,"domainLookupStart":609,"fetchStart":609,"redirectEnd":0,"redirectStart":0,"requestStart":609,"responseEnd":1170.5,"responseStart":1170.5,"secureConnectionStart":609},{"duration":650.3000001907349,"initiatorType":"link","name":"https://jira.mariadb.org/s/b04b06a02d1959df322d9cded3aeecc1-CDN/lu2cib/820016/12ta74/a2ff6aa845ffc9a1d22fe23d9ee791fc/_/download/contextbatch/css/jira.global.look-and-feel,-_super/batch.css","startTime":609.3000001907349,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":609.3000001907349,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":1259.6000003814697,"responseStart":0,"secureConnectionStart":0},{"duration":561.5999994277954,"initiatorType":"script","name":"https://jira.mariadb.org/rest/api/1.0/shortcuts/820016/47140b6e0a9bc2e4913da06536125810/shortcuts.js?context=issuenavigation&context=issueaction","startTime":609.4000005722046,"connectEnd":609.4000005722046,"connectStart":609.4000005722046,"domainLookupEnd":609.4000005722046,"domainLookupStart":609.4000005722046,"fetchStart":609.4000005722046,"redirectEnd":0,"redirectStart":0,"requestStart":609.4000005722046,"responseEnd":1171,"responseStart":1171,"secureConnectionStart":609.4000005722046},{"duration":650.0999994277954,"initiatorType":"link","name":"https://jira.mariadb.org/s/3ac36323ba5e4eb0af2aa7ac7211b4bb-CDN/lu2cib/820016/12ta74/d176f0986478cc64f24226b3d20c140d/_/download/contextbatch/css/com.atlassian.jira.projects.sidebar.init,-_super,-project.issue.navigator,-jira.view.issue/batch.css?jira.create.linked.issue=true","startTime":609.6000003814697,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":609.6000003814697,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":1259.6999998092651,"responseStart":0,"secureConnectionStart":0},{"duration":562,"initiatorType":"script","name":"https://jira.mariadb.org/s/5d5e8fe91fbc506585e83ea3b62ccc4b-CDN/lu2cib/820016/12ta74/d176f0986478cc64f24226b3d20c140d/_/download/contextbatch/js/com.atlassian.jira.projects.sidebar.init,-_super,-project.issue.navigator,-jira.view.issue/batch.js?jira.create.linked.issue=true&locale=en","startTime":609.6999998092651,"connectEnd":609.6999998092651,"connectStart":609.6999998092651,"domainLookupEnd":609.6999998092651,"domainLookupStart":609.6999998092651,"fetchStart":609.6999998092651,"redirectEnd":0,"redirectStart":0,"requestStart":609.6999998092651,"responseEnd":1171.6999998092651,"responseStart":1171.6999998092651,"secureConnectionStart":609.6999998092651},{"duration":902,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/1.0/_/download/batch/jira.webresources:bigpipe-js/jira.webresources:bigpipe-js.js","startTime":610.8000001907349,"connectEnd":610.8000001907349,"connectStart":610.8000001907349,"domainLookupEnd":610.8000001907349,"domainLookupStart":610.8000001907349,"fetchStart":610.8000001907349,"redirectEnd":0,"redirectStart":0,"requestStart":610.8000001907349,"responseEnd":1512.8000001907349,"responseStart":1512.8000001907349,"secureConnectionStart":610.8000001907349},{"duration":1306.3000001907349,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/1.0/_/download/batch/jira.webresources:bigpipe-init/jira.webresources:bigpipe-init.js","startTime":616.6000003814697,"connectEnd":616.6000003814697,"connectStart":616.6000003814697,"domainLookupEnd":616.6000003814697,"domainLookupStart":616.6000003814697,"fetchStart":616.6000003814697,"redirectEnd":0,"redirectStart":0,"requestStart":616.6000003814697,"responseEnd":1922.9000005722046,"responseStart":1922.9000005722046,"secureConnectionStart":616.6000003814697},{"duration":252.29999923706055,"initiatorType":"xmlhttprequest","name":"https://jira.mariadb.org/rest/webResources/1.0/resources","startTime":1271.4000005722046,"connectEnd":1271.4000005722046,"connectStart":1271.4000005722046,"domainLookupEnd":1271.4000005722046,"domainLookupStart":1271.4000005722046,"fetchStart":1271.4000005722046,"redirectEnd":0,"redirectStart":0,"requestStart":1271.4000005722046,"responseEnd":1523.6999998092651,"responseStart":1523.6999998092651,"secureConnectionStart":1271.4000005722046},{"duration":430.30000019073486,"initiatorType":"link","name":"https://jira.mariadb.org/s/d5715adaadd168a9002b108b2b039b50-CDN/lu2cib/820016/12ta74/be4b45e9cec53099498fa61c8b7acba4/_/download/contextbatch/css/jira.project.sidebar,-_super,-project.issue.navigator,-jira.general,-jira.browse.project,-jira.view.issue,-jira.global,-atl.general,-com.atlassian.jira.projects.sidebar.init/batch.css?agile_global_admin_condition=true&jag=true&jira.create.linked.issue=true&slack-enabled=true&whisper-enabled=true","startTime":1537.6999998092651,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":1537.6999998092651,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":1968,"responseStart":0,"secureConnectionStart":0},{"duration":394.1000003814697,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/e65b778d185daf5aee24936755b43da6/_/download/contextbatch/js/browser-metrics-plugin.contrib,-_super,-project.issue.navigator,-jira.view.issue,-atl.general/batch.js?agile_global_admin_condition=true&jag=true&jira.create.linked.issue=true&slack-enabled=true&whisper-enabled=true","startTime":1538.6999998092651,"connectEnd":1538.6999998092651,"connectStart":1538.6999998092651,"domainLookupEnd":1538.6999998092651,"domainLookupStart":1538.6999998092651,"fetchStart":1538.6999998092651,"redirectEnd":0,"redirectStart":0,"requestStart":1538.6999998092651,"responseEnd":1932.8000001907349,"responseStart":1932.8000001907349,"secureConnectionStart":1538.6999998092651},{"duration":400.5,"initiatorType":"script","name":"https://jira.mariadb.org/s/097ae97cb8fbec7d6ea4bbb1f26955b9-CDN/lu2cib/820016/12ta74/be4b45e9cec53099498fa61c8b7acba4/_/download/contextbatch/js/jira.project.sidebar,-_super,-project.issue.navigator,-jira.general,-jira.browse.project,-jira.view.issue,-jira.global,-atl.general,-com.atlassian.jira.projects.sidebar.init/batch.js?agile_global_admin_condition=true&jag=true&jira.create.linked.issue=true&locale=en&slack-enabled=true&whisper-enabled=true","startTime":1539.1000003814697,"connectEnd":1539.1000003814697,"connectStart":1539.1000003814697,"domainLookupEnd":1539.1000003814697,"domainLookupStart":1539.1000003814697,"fetchStart":1539.1000003814697,"redirectEnd":0,"redirectStart":0,"requestStart":1539.1000003814697,"responseEnd":1939.6000003814697,"responseStart":1939.6000003814697,"secureConnectionStart":1539.1000003814697},{"duration":402,"initiatorType":"script","name":"https://www.google-analytics.com/analytics.js","startTime":1581,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":1581,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":1983,"responseStart":0,"secureConnectionStart":0}],"fetchStart":0,"domainLookupStart":0,"domainLookupEnd":0,"connectStart":0,"connectEnd":0,"requestStart":370,"responseStart":601,"responseEnd":611,"domLoading":604,"domInteractive":2027,"domContentLoadedEventStart":2027,"domContentLoadedEventEnd":2095,"domComplete":2360,"loadEventStart":2360,"loadEventEnd":2361,"userAgent":"Mozilla/5.0 AppleWebKit/537.36 (KHTML, like Gecko; compatible; ClaudeBot/1.0; +claudebot@anthropic.com)","marks":[{"name":"bigPipe.sidebar-id.start","time":1924.6000003814697},{"name":"bigPipe.sidebar-id.end","time":1925.5},{"name":"bigPipe.activity-panel-pipe-id.start","time":1925.6999998092651},{"name":"bigPipe.activity-panel-pipe-id.end","time":1934.5},{"name":"activityTabFullyLoaded","time":2125.1000003814697}],"measures":[],"correlationId":"d1e04b81f33acc","effectiveType":"4g","downlink":10,"rtt":0,"serverDuration":165,"dbReadsTimeInMs":20,"dbConnsTimeInMs":30,"applicationHash":"9d11dbea5f4be3d4cc21f03a88dd11d8c8687422","experiments":[]}}
bb-10.6-mdev34720 has a patch that is a work-in-progress