Commit 6d6a8df9 authored by Adam Cohen's avatar Adam Cohen Committed by Dylan Griffith

Document allow_cross_joins_across_databases

parent 6ead7b93
...@@ -98,22 +98,45 @@ and their tables must be placed in two directories for now: ...@@ -98,22 +98,45 @@ and their tables must be placed in two directories for now:
We aim to keep the schema for both tables the same across both databases. We aim to keep the schema for both tables the same across both databases.
<!--
NOTE: The `validate_cross_joins!` method in `spec/support/database/prevent_cross_joins.rb` references
the following heading in the code, so if you make a change to this heading, make sure to update
the corresponding documentation URL used in `spec/support/database/prevent_cross_joins.rb`.
-->
### Removing joins between `ci_*` and non `ci_*` tables ### Removing joins between `ci_*` and non `ci_*` tables
We are planning on moving all the `ci_*` tables to a separate database so Queries that join across databases raise an error. [Introduced](https://gitlab.com/gitlab-org/gitlab/-/merge_requests/68620)
in GitLab 14.3, for new queries only. Pre-existing queries do not raise an error.
We are planning on moving all the `ci_*` tables to a separate database, so
referencing `ci_*` tables with other tables will not be possible. This means, referencing `ci_*` tables with other tables will not be possible. This means,
that using any kind of `JOIN` in SQL queries will not work. We have identified that using any kind of `JOIN` in SQL queries will not work. We have identified
already many such examples that need to be fixed in already many such examples that need to be fixed in
<https://gitlab.com/groups/gitlab-org/-/epics/6289> . <https://gitlab.com/groups/gitlab-org/-/epics/6289> .
The following are some real examples that have resulted from this and these #### Path to removing cross-database joins
patterns may apply to future cases.
The following steps are the process to remove cross-database joins between
`ci_*` and non `ci_*` tables:
1. **{check-circle}** Add all failing specs to the [`cross-join-allowlist.yml`](https://gitlab.com/gitlab-org/gitlab/-/blob/f5de89daeb468fc45e1e95a76d1b5297aa53da11/spec/support/database/cross-join-allowlist.yml)
file.
1. **{dotted-circle}** Find the code that caused the spec failure and wrap the isolated code
in [`allow_cross_joins_across_databases`](#allowlist-for-existing-cross-joins).
Link to a new issue assigned to the correct team to remove the specs from the
`cross-join-allowlist.yml` file.
1. **{dotted-circle}** Remove the `cross-join-allowlist.yml` file and stop allowing
whole test files.
1. **{dotted-circle}** Fix the problem and remove the `allow_cross_joins_across_databases` call.
1. **{dotted-circle}** Fix all the cross-joins and remove the `allow_cross_joins_across_databases` method.
[Introduced](https://gitlab.com/gitlab-org/gitlab/-/merge_requests/68620) in GitLab 14.3, any #### Suggestions for removing cross-database joins
queries detected that join across databases raises an error (except
for pre-existing queries).
#### Remove the code The following sections are some real examples that were identified as joining across databases,
along with possible suggestions on how to fix them.
##### Remove the code
The simplest solution we've seen several times now has been an existing scope The simplest solution we've seen several times now has been an existing scope
that is unused. This is the easiest example to fix. So the first step is to that is unused. This is the easiest example to fix. So the first step is to
...@@ -135,7 +158,7 @@ to evaluate, because `UsageData` is not critical to users and it may be possible ...@@ -135,7 +158,7 @@ to evaluate, because `UsageData` is not critical to users and it may be possible
to get a similarly useful metric with a simpler approach. Alternatively we may to get a similarly useful metric with a simpler approach. Alternatively we may
find that nobody is using these metrics, so we can remove them. find that nobody is using these metrics, so we can remove them.
#### Use `preload` instead of `includes` ##### Use `preload` instead of `includes`
The `includes` and `preload` methods in Rails are both ways to avoid an N+1 The `includes` and `preload` methods in Rails are both ways to avoid an N+1
query. The `includes` method in Rails uses a heuristic approach to determine query. The `includes` method in Rails uses a heuristic approach to determine
...@@ -149,7 +172,7 @@ allows you to avoid the join, while still avoiding the N+1 query. ...@@ -149,7 +172,7 @@ allows you to avoid the join, while still avoiding the N+1 query.
You can see a real example of this solution being used in You can see a real example of this solution being used in
<https://gitlab.com/gitlab-org/gitlab/-/merge_requests/67655>. <https://gitlab.com/gitlab-org/gitlab/-/merge_requests/67655>.
#### De-normalize some foreign key to the table ##### De-normalize some foreign key to the table
De-normalization refers to adding redundant precomputed (duplicated) data to De-normalization refers to adding redundant precomputed (duplicated) data to
a table to simplify certain queries or to improve performance. In this a table to simplify certain queries or to improve performance. In this
...@@ -202,7 +225,7 @@ You can see this approach implemented in ...@@ -202,7 +225,7 @@ You can see this approach implemented in
<https://gitlab.com/gitlab-org/gitlab/-/merge_requests/66963> . This MR also <https://gitlab.com/gitlab-org/gitlab/-/merge_requests/66963> . This MR also
de-normalizes `pipeline_id` to fix a similar query. de-normalizes `pipeline_id` to fix a similar query.
#### De-normalize into an extra table ##### De-normalize into an extra table
Sometimes the previous de-normalization (adding an extra column) doesn't work for Sometimes the previous de-normalization (adding an extra column) doesn't work for
your specific case. This may be due to the fact that your data is not 1:1, or your specific case. This may be due to the fact that your data is not 1:1, or
...@@ -249,7 +272,7 @@ logic to delete these rows if or whenever necessary in your domain. ...@@ -249,7 +272,7 @@ logic to delete these rows if or whenever necessary in your domain.
Finally, this de-normalization and new query also improves performance because Finally, this de-normalization and new query also improves performance because
it does less joins and needs less filtering. it does less joins and needs less filtering.
#### Use `disable_joins` for `has_one` or `has_many` `through:` relations ##### Use `disable_joins` for `has_one` or `has_many` `through:` relations
Sometimes a join query is caused by using `has_one ... through:` or `has_many Sometimes a join query is caused by using `has_one ... through:` or `has_many
... through:` across tables that span the different databases. These joins ... through:` across tables that span the different databases. These joins
...@@ -365,3 +388,27 @@ end ...@@ -365,3 +388,27 @@ end
You can see a real example of using this method for fixing a cross-join in You can see a real example of using this method for fixing a cross-join in
<https://gitlab.com/gitlab-org/gitlab/-/merge_requests/67655>. <https://gitlab.com/gitlab-org/gitlab/-/merge_requests/67655>.
#### Allowlist for existing cross-joins
A cross-join across databases can be explicitly allowed by wrapping the code in the
`::Gitlab::Database.allow_cross_joins_across_databases` helper method.
This method should only be used:
- For existing code.
- If the code is required to help migrate away from a cross-join. For example,
in a migration that backfills data for future use to remove a cross-join.
The `allow_cross_joins_across_databases` helper method can be used as follows:
```ruby
::Gitlab::Database.allow_cross_joins_across_databases(url: 'https://gitlab.com/gitlab-org/gitlab/-/issues/336590') do
subject.perform(1, 4)
end
```
The `url` parameter should point to an issue with a milestone for when we intend
to fix the cross-join. If the cross-join is being used in a migration, we do not
need to fix the code. See <https://gitlab.com/gitlab-org/gitlab/-/issues/340017>
for more details.
...@@ -47,7 +47,7 @@ module Database ...@@ -47,7 +47,7 @@ module Database
Thread.current[:has_cross_join_exception] = true Thread.current[:has_cross_join_exception] = true
raise CrossJoinAcrossUnsupportedTablesError, raise CrossJoinAcrossUnsupportedTablesError,
"Unsupported cross-join across '#{tables.join(", ")}' modifying '#{schemas.to_a.join(", ")}' discovered " \ "Unsupported cross-join across '#{tables.join(", ")}' modifying '#{schemas.to_a.join(", ")}' discovered " \
"when executing query '#{sql}'" "when executing query '#{sql}'. Please refer to https://docs.gitlab.com/ee/development/database/multiple_databases.html#removing-joins-between-ci_-and-non-ci_-tables for details on how to resolve this exception."
end end
end end
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment