Currently InnoDB uses internal parser for adding foreign keys. Remove internal parser and use data parsed by SQL parser (sql_yacc) for adding foreign keys. This allows to pass some additional data from SQL layer like whether the referenced table is SYSTEM_TIME partitioned (MDEV-19191) and is required for further improvements (MDEV-16417, MDEV-10393, MDEV-12483).
I think that as part of this, we should fix the SQL parser (both sql_yacc.yy and sql_yacc_ora.yy) so that also the following form of REFERENCES constraints will be recognized and not ignored by the parser:
column_def:
field_spec
{ $$= $1; }
| field_spec references
{ $$= $1; }
;
Currently, the grammar rule is ignoring the semantic value of the REFERENCES clause. The non-terminal references is being more correctly used in the key_def rule. Also, I wonder whether the second production should include opt_constraint:
column_def:
field_spec
{ $$= $1; }
| field_spec opt_constraint references
{ $$= … $1 … $2 … $3 …; }
;
Marko Mäkelä
added a comment - I think that as part of this, we should fix the SQL parser (both sql_yacc.yy and sql_yacc_ora.yy ) so that also the following form of REFERENCES constraints will be recognized and not ignored by the parser:
column_def:
field_spec
{ $$= $1; }
| field_spec references
{ $$= $1; }
;
Currently, the grammar rule is ignoring the semantic value of the REFERENCES clause. The non-terminal references is being more correctly used in the key_def rule. Also, I wonder whether the second production should include opt_constraint :
column_def:
field_spec
{ $$= $1; }
| field_spec opt_constraint references
{ $$= … $1 … $2 … $3 …; }
;
bar thinks and I agree that the FOREIGN KEY metadata would better be stored in TABLE_SHARE. Sure, it would still not be stored anywhere outside InnoDB (until MDEV-16417 is implemented), but TABLE_SHARE would be the logical place for it.
Note: Until MDEV-16417, we will need some interface that allows the storage engine to fill in the FOREIGN KEY metadata. That might be some kind of an extension to handler::open(). This is a bit tricky, because there could be multiple concurrent calls to that function for the same table, while at the same time there was no prior call of handler::open() for that table. Inside ha_innobase::open() there is some distinction of ‘first-time open’ and ‘reopen’. If the ‘first-time open’ would initialize the FOREIGN KEY metadata in TABLE_SHARE, we would need some locking to make sure that the concurrent calls that would reach the ‘reopen’ code path will not return before that metadata has been initialized.
I sent some lower-level review comments on GitHub.
Please rebase the work on the latest 10.5 and remember to update the commit date when updating the patch.
Marko Mäkelä
added a comment - bar thinks and I agree that the FOREIGN KEY metadata would better be stored in TABLE_SHARE . Sure, it would still not be stored anywhere outside InnoDB (until MDEV-16417 is implemented), but TABLE_SHARE would be the logical place for it.
Note: Until MDEV-16417 , we will need some interface that allows the storage engine to fill in the FOREIGN KEY metadata. That might be some kind of an extension to handler::open() . This is a bit tricky, because there could be multiple concurrent calls to that function for the same table, while at the same time there was no prior call of handler::open() for that table. Inside ha_innobase::open() there is some distinction of ‘first-time open’ and ‘reopen’. If the ‘first-time open’ would initialize the FOREIGN KEY metadata in TABLE_SHARE , we would need some locking to make sure that the concurrent calls that would reach the ‘reopen’ code path will not return before that metadata has been initialized.
I sent some lower-level review comments on GitHub.
Please rebase the work on the latest 10.5 and remember to update the commit date when updating the patch.
Refactor InnoDB to use foreign keys from TABLE_SHARE is large subtask of MDEV-16417. There are at least 16 functions that need to be remade or refactored including ~35 occurrences of `SYS_FOREIGN` and ~27 occurrences of `SYS_FOREIGN_COLS` in InnoDB SQL code.
What is possible in scope of this task is to pass via `TABLE_SHARE` foreign keys *only* for `create_table_info_t::create_table()`. There is no much importance in that for now as there is raw data in `alter_info` and there is processed data in `TABLE_SHARE` for `create_table_info_t::create_table()`, so we just switch from one to another. And there is no final design of foreign keys structure in `TABLE_SHARE` until MDEV-16417, so IMO passing them now via `TABLE_SHARE` is a bit premature.
Aleksey Midenkov
added a comment - - edited Refactor InnoDB to use foreign keys from TABLE_SHARE is large subtask of MDEV-16417 . There are at least 16 functions that need to be remade or refactored including ~35 occurrences of `SYS_FOREIGN` and ~27 occurrences of `SYS_FOREIGN_COLS` in InnoDB SQL code.
What is possible in scope of this task is to pass via `TABLE_SHARE` foreign keys * only * for `create_table_info_t::create_table()`. There is no much importance in that for now as there is raw data in `alter_info` and there is processed data in `TABLE_SHARE` for `create_table_info_t::create_table()`, so we just switch from one to another. And there is no final design of foreign keys structure in `TABLE_SHARE` until MDEV-16417 , so IMO passing them now via `TABLE_SHARE` is a bit premature.
midenok, you mean, refactor InnoDB to use foreign keys from TABLE_SHARE? I don't think this is particularly important. InnoDB has dict_table_t and dict_col_t and dict_index_t despite the fact that table metadata, and all columns and indexes are stored in the TABLE_SHARE too.
But the server does not use InnoDB data structures, the server uses what's in the TABLE_SHARE. Only for foreign keys the server asks InnoDB for FK information. Instead it should store all FK metadata in the TABLE_SHARE and use them from there.
While an engine can still store them internally and not peek into TABLE_SHARE every time.
Sergei Golubchik
added a comment - midenok , you mean, refactor InnoDB to use foreign keys from TABLE_SHARE ? I don't think this is particularly important. InnoDB has dict_table_t and dict_col_t and dict_index_t despite the fact that table metadata, and all columns and indexes are stored in the TABLE_SHARE too.
But the server does not use InnoDB data structures, the server uses what's in the TABLE_SHARE . Only for foreign keys the server asks InnoDB for FK information. Instead it should store all FK metadata in the TABLE_SHARE and use them from there.
While an engine can still store them internally and not peek into TABLE_SHARE every time.
This additional subtask requires several days to finish as it is required to keep in sync referenced TABLE_SHARE objects (TABLE_SHARE::referenced_keys list) after DROP TABLE, ALTER ADD/DROP FOREIGN KEY, RENAME TABLE. It was expected to be a subtask of MDEV-16417, but I already started it here and implemented a draft for RENAME TABLE: https://github.com/MariaDB/server/commit/abcce38a623d86bdd9e27cfd0b04ae9698cfef0b
And the above fix should be remade for avoiding TABLE_SHARE invalidation.
P.S. Actually this big enough subtask took more than several days.
Aleksey Midenkov
added a comment - - edited This additional subtask requires several days to finish as it is required to keep in sync referenced TABLE_SHARE objects (TABLE_SHARE::referenced_keys list) after DROP TABLE, ALTER ADD/DROP FOREIGN KEY, RENAME TABLE. It was expected to be a subtask of MDEV-16417 , but I already started it here and implemented a draft for RENAME TABLE: https://github.com/MariaDB/server/commit/abcce38a623d86bdd9e27cfd0b04ae9698cfef0b
And the above fix should be remade for avoiding TABLE_SHARE invalidation.
MDEV-20865 will track further progress.
P.S. Actually this big enough subtask took more than several days.
People
Aleksey Midenkov
Aleksey Midenkov
Votes:
1Vote for this issue
Watchers:
7Start 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":871.3000001907349,"ttfb":259.40000009536743,"pageVisibility":"visible","entityId":78682,"key":"jira.project.issue.view-issue","isInitial":true,"threshold":1000,"elementTimings":{},"userDeviceMemory":8,"userDeviceProcessors":64,"apdex":1,"journeyId":"f1338c65-b01e-4a1e-83d3-38b116cbbe37","navigationType":0,"readyForUser":950.3000001907349,"redirectCount":0,"resourceLoadedEnd":908.7000000476837,"resourceLoadedStart":287.90000009536743,"resourceTiming":[{"duration":101,"initiatorType":"link","name":"https://jira.mariadb.org/s/2c21342762a6a02add1c328bed317ffd-CDN/lu2cib/820016/12ta74/0a8bac35585be7fc6c9cc5a0464cd4cf/_/download/contextbatch/css/_super/batch.css","startTime":287.90000009536743,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":287.90000009536743,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":388.90000009536743,"responseStart":0,"secureConnectionStart":0},{"duration":101,"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":288.10000014305115,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":288.10000014305115,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":389.10000014305115,"responseStart":0,"secureConnectionStart":0},{"duration":109.59999990463257,"initiatorType":"script","name":"https://jira.mariadb.org/s/0917945aaa57108d00c5076fea35e069-CDN/lu2cib/820016/12ta74/0a8bac35585be7fc6c9cc5a0464cd4cf/_/download/contextbatch/js/_super/batch.js?locale=en","startTime":288.30000019073486,"connectEnd":288.30000019073486,"connectStart":288.30000019073486,"domainLookupEnd":288.30000019073486,"domainLookupStart":288.30000019073486,"fetchStart":288.30000019073486,"redirectEnd":0,"redirectStart":0,"requestStart":288.30000019073486,"responseEnd":397.90000009536743,"responseStart":397.90000009536743,"secureConnectionStart":288.30000019073486},{"duration":249.90000009536743,"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":288.5,"connectEnd":288.5,"connectStart":288.5,"domainLookupEnd":288.5,"domainLookupStart":288.5,"fetchStart":288.5,"redirectEnd":0,"redirectStart":0,"requestStart":288.5,"responseEnd":538.4000000953674,"responseStart":538.4000000953674,"secureConnectionStart":288.5},{"duration":253.5,"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":288.7000000476837,"connectEnd":288.7000000476837,"connectStart":288.7000000476837,"domainLookupEnd":288.7000000476837,"domainLookupStart":288.7000000476837,"fetchStart":288.7000000476837,"redirectEnd":0,"redirectStart":0,"requestStart":288.7000000476837,"responseEnd":542.2000000476837,"responseStart":542.2000000476837,"secureConnectionStart":288.7000000476837},{"duration":253.70000004768372,"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":288.90000009536743,"connectEnd":288.90000009536743,"connectStart":288.90000009536743,"domainLookupEnd":288.90000009536743,"domainLookupStart":288.90000009536743,"fetchStart":288.90000009536743,"redirectEnd":0,"redirectStart":0,"requestStart":288.90000009536743,"responseEnd":542.6000001430511,"responseStart":542.6000001430511,"secureConnectionStart":288.90000009536743},{"duration":253.89999985694885,"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":289.10000014305115,"connectEnd":289.10000014305115,"connectStart":289.10000014305115,"domainLookupEnd":289.10000014305115,"domainLookupStart":289.10000014305115,"fetchStart":289.10000014305115,"redirectEnd":0,"redirectStart":0,"requestStart":289.10000014305115,"responseEnd":543,"responseStart":543,"secureConnectionStart":289.10000014305115},{"duration":256.10000014305115,"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":289.2000000476837,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":289.2000000476837,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":545.3000001907349,"responseStart":0,"secureConnectionStart":0},{"duration":254.19999980926514,"initiatorType":"script","name":"https://jira.mariadb.org/rest/api/1.0/shortcuts/820016/47140b6e0a9bc2e4913da06536125810/shortcuts.js?context=issuenavigation&context=issueaction","startTime":289.30000019073486,"connectEnd":289.30000019073486,"connectStart":289.30000019073486,"domainLookupEnd":289.30000019073486,"domainLookupStart":289.30000019073486,"fetchStart":289.30000019073486,"redirectEnd":0,"redirectStart":0,"requestStart":289.30000019073486,"responseEnd":543.5,"responseStart":543.5,"secureConnectionStart":289.30000019073486},{"duration":255.90000009536743,"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":289.5,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":289.5,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":545.4000000953674,"responseStart":0,"secureConnectionStart":0},{"duration":265.89999985694885,"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":289.60000014305115,"connectEnd":289.60000014305115,"connectStart":289.60000014305115,"domainLookupEnd":289.60000014305115,"domainLookupStart":289.60000014305115,"fetchStart":289.60000014305115,"redirectEnd":0,"redirectStart":0,"requestStart":289.60000014305115,"responseEnd":555.5,"responseStart":555.5,"secureConnectionStart":289.60000014305115},{"duration":508.7999999523163,"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":291.30000019073486,"connectEnd":291.30000019073486,"connectStart":291.30000019073486,"domainLookupEnd":291.30000019073486,"domainLookupStart":291.30000019073486,"fetchStart":291.30000019073486,"redirectEnd":0,"redirectStart":0,"requestStart":291.30000019073486,"responseEnd":800.1000001430511,"responseStart":800.1000001430511,"secureConnectionStart":291.30000019073486},{"duration":591.5,"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":291.30000019073486,"connectEnd":291.30000019073486,"connectStart":291.30000019073486,"domainLookupEnd":291.30000019073486,"domainLookupStart":291.30000019073486,"fetchStart":291.30000019073486,"redirectEnd":0,"redirectStart":0,"requestStart":291.30000019073486,"responseEnd":882.8000001907349,"responseStart":882.8000001907349,"secureConnectionStart":291.30000019073486},{"duration":146.70000004768372,"initiatorType":"xmlhttprequest","name":"https://jira.mariadb.org/rest/webResources/1.0/resources","startTime":549,"connectEnd":549,"connectStart":549,"domainLookupEnd":549,"domainLookupStart":549,"fetchStart":549,"redirectEnd":0,"redirectStart":0,"requestStart":549,"responseEnd":695.7000000476837,"responseStart":695.7000000476837,"secureConnectionStart":549},{"duration":75.79999995231628,"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":809.7000000476837,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":809.7000000476837,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":885.5,"responseStart":0,"secureConnectionStart":0},{"duration":91.70000004768372,"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":810.7000000476837,"connectEnd":810.7000000476837,"connectStart":810.7000000476837,"domainLookupEnd":810.7000000476837,"domainLookupStart":810.7000000476837,"fetchStart":810.7000000476837,"redirectEnd":0,"redirectStart":0,"requestStart":810.7000000476837,"responseEnd":902.4000000953674,"responseStart":902.4000000953674,"secureConnectionStart":810.7000000476837},{"duration":97.70000004768372,"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":811,"connectEnd":811,"connectStart":811,"domainLookupEnd":811,"domainLookupStart":811,"fetchStart":811,"redirectEnd":0,"redirectStart":0,"requestStart":811,"responseEnd":908.7000000476837,"responseStart":908.7000000476837,"secureConnectionStart":811},{"duration":256.2000000476837,"initiatorType":"script","name":"https://www.google-analytics.com/analytics.js","startTime":865,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":865,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":1121.2000000476837,"responseStart":0,"secureConnectionStart":0}],"fetchStart":0,"domainLookupStart":0,"domainLookupEnd":0,"connectStart":0,"connectEnd":0,"requestStart":69,"responseStart":260,"responseEnd":265,"domLoading":285,"domInteractive":1061,"domContentLoadedEventStart":1061,"domContentLoadedEventEnd":1111,"domComplete":1328,"loadEventStart":1328,"loadEventEnd":1328,"userAgent":"Mozilla/5.0 AppleWebKit/537.36 (KHTML, like Gecko; compatible; ClaudeBot/1.0; +claudebot@anthropic.com)","marks":[{"name":"bigPipe.sidebar-id.start","time":1037.4000000953674},{"name":"bigPipe.sidebar-id.end","time":1038.4000000953674},{"name":"bigPipe.activity-panel-pipe-id.start","time":1038.5},{"name":"bigPipe.activity-panel-pipe-id.end","time":1041.8000001907349},{"name":"activityTabFullyLoaded","time":1134.6000001430511}],"measures":[],"correlationId":"29a7bb740ea8af","effectiveType":"4g","downlink":9.1,"rtt":0,"serverDuration":133,"dbReadsTimeInMs":34,"dbConnsTimeInMs":47,"applicationHash":"9d11dbea5f4be3d4cc21f03a88dd11d8c8687422","experiments":[]}}
I think that as part of this, we should fix the SQL parser (both sql_yacc.yy and sql_yacc_ora.yy) so that also the following form of REFERENCES constraints will be recognized and not ignored by the parser:
column_def:
field_spec
{ $$= $1; }
| field_spec references
{ $$= $1; }
;
Currently, the grammar rule is ignoring the semantic value of the REFERENCES clause. The non-terminal references is being more correctly used in the key_def rule. Also, I wonder whether the second production should include opt_constraint:
column_def:
field_spec
{ $$= $1; }
| field_spec opt_constraint references
{ $$= … $1 … $2 … $3 …; }
;