debian-mirror-gitlab/lib/gitlab/database/gitlab_schema.rb

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

167 lines
5.9 KiB
Ruby
Raw Normal View History

2021-12-11 22:18:48 +05:30
# frozen_string_literal: true
# This module gathers information about table to schema mapping
# to understand table affinity
#
# Each table / view needs to have assigned gitlab_schema. Names supported today:
#
# - gitlab_shared - defines a set of tables that are found on all databases (data accessed is dependent on connection)
2023-03-04 22:38:38 +05:30
# - gitlab_main / gitlab_ci - defines a set of tables that can only exist on a given application database
# - gitlab_geo - defines a set of tables that can only exist on the geo database
# - gitlab_internal - defines all internal tables of Rails and PostgreSQL
2021-12-11 22:18:48 +05:30
#
# Tables for the purpose of tests should be prefixed with `_test_my_table_name`
module Gitlab
module Database
module GitlabSchema
2023-03-17 16:20:25 +05:30
UnknownSchemaError = Class.new(StandardError)
2023-03-04 22:38:38 +05:30
DICTIONARY_PATH = 'db/docs/'
2022-08-13 15:12:31 +05:30
2023-07-09 08:55:56 +05:30
def self.table_schemas!(tables)
tables.map { |table| table_schema!(table) }.to_set
2021-12-11 22:18:48 +05:30
end
2023-07-09 08:55:56 +05:30
# rubocop:disable Metrics/CyclomaticComplexity
def self.table_schema(name)
2021-12-11 22:18:48 +05:30
schema_name, table_name = name.split('.', 2) # Strip schema name like: `public.`
# Most of names do not have schemas, ensure that this is table
unless table_name
table_name = schema_name
schema_name = nil
end
# strip partition number of a form `loose_foreign_keys_deleted_records_1`
table_name.gsub!(/_[0-9]+$/, '')
2023-03-17 16:20:25 +05:30
# Tables and views that are properly mapped
2023-03-04 22:38:38 +05:30
if gitlab_schema = views_and_tables_to_schema[table_name]
2021-12-11 22:18:48 +05:30
return gitlab_schema
end
2023-03-17 16:20:25 +05:30
# Tables and views that are deleted, but we still need to reference them
if gitlab_schema = deleted_views_and_tables_to_schema[table_name]
2021-12-11 22:18:48 +05:30
return gitlab_schema
end
2023-05-27 22:25:52 +05:30
# Partitions that belong to the CI domain
if table_name.start_with?('ci_') && gitlab_schema = views_and_tables_to_schema["p_#{table_name}"]
return gitlab_schema
end
2022-07-23 23:45:48 +05:30
# All tables from `information_schema.` are marked as `internal`
return :gitlab_internal if schema_name == 'information_schema'
2021-12-11 22:18:48 +05:30
2022-04-04 11:22:00 +05:30
return :gitlab_main if table_name.start_with?('_test_gitlab_main_')
return :gitlab_ci if table_name.start_with?('_test_gitlab_ci_')
2023-07-09 08:55:56 +05:30
return :gitlab_embedding if table_name.start_with?('_test_gitlab_embedding_')
2023-03-04 22:38:38 +05:30
return :gitlab_geo if table_name.start_with?('_test_gitlab_geo_')
2022-04-04 11:22:00 +05:30
# All tables that start with `_test_` without a following schema are shared and ignored
2021-12-11 22:18:48 +05:30
return :gitlab_shared if table_name.start_with?('_test_')
2022-07-23 23:45:48 +05:30
# All `pg_` tables are marked as `internal`
return :gitlab_internal if table_name.start_with?('pg_')
2021-12-11 22:18:48 +05:30
2023-07-09 08:55:56 +05:30
# Sometimes the name of an index can be interpreted as a table's name.
# For eg, if we execute "ALTER INDEX my_index..", my_index is interpreted as a table name.
# In such cases, we should return the schema of the database table actually
# holding that index.
index_name = table_name
derive_schema_from_index(index_name)
2023-03-04 22:38:38 +05:30
end
2023-07-09 08:55:56 +05:30
# rubocop:enable Metrics/CyclomaticComplexity
2023-03-04 22:38:38 +05:30
def self.dictionary_path_globs
[Rails.root.join(DICTIONARY_PATH, '*.yml')]
end
def self.view_path_globs
[Rails.root.join(DICTIONARY_PATH, 'views', '*.yml')]
end
2023-03-17 16:20:25 +05:30
def self.deleted_views_path_globs
[Rails.root.join(DICTIONARY_PATH, 'deleted_views', '*.yml')]
end
def self.deleted_tables_path_globs
[Rails.root.join(DICTIONARY_PATH, 'deleted_tables', '*.yml')]
end
2023-03-04 22:38:38 +05:30
def self.views_and_tables_to_schema
@views_and_tables_to_schema ||= self.tables_to_schema.merge(self.views_to_schema)
2021-12-11 22:18:48 +05:30
end
2023-03-17 16:20:25 +05:30
def self.table_schema!(name)
2023-07-09 08:55:56 +05:30
# rubocop:disable Gitlab/DocUrl
self.table_schema(name) || raise(
2023-03-17 16:20:25 +05:30
UnknownSchemaError,
2023-07-09 08:55:56 +05:30
"Could not find gitlab schema for table #{name}: Any new or deleted tables must be added to the database dictionary " \
"See https://docs.gitlab.com/ee/development/database/database_dictionary.html"
2023-03-17 16:20:25 +05:30
)
2023-07-09 08:55:56 +05:30
# rubocop:enable Gitlab/DocUrl
2023-03-17 16:20:25 +05:30
end
2023-03-04 22:38:38 +05:30
2023-03-17 16:20:25 +05:30
def self.deleted_views_and_tables_to_schema
@deleted_views_and_tables_to_schema ||= self.deleted_tables_to_schema.merge(self.deleted_views_to_schema)
2023-03-04 22:38:38 +05:30
end
2023-03-17 16:20:25 +05:30
def self.deleted_tables_to_schema
@deleted_tables_to_schema ||= self.build_dictionary(self.deleted_tables_path_globs)
end
2023-03-04 22:38:38 +05:30
2023-03-17 16:20:25 +05:30
def self.deleted_views_to_schema
@deleted_views_to_schema ||= self.build_dictionary(self.deleted_views_path_globs)
end
def self.tables_to_schema
@tables_to_schema ||= self.build_dictionary(self.dictionary_path_globs)
end
def self.views_to_schema
@views_to_schema ||= self.build_dictionary(self.view_path_globs)
2021-12-11 22:18:48 +05:30
end
2022-05-07 20:08:51 +05:30
def self.schema_names
2023-03-04 22:38:38 +05:30
@schema_names ||= self.views_and_tables_to_schema.values.to_set
2022-05-07 20:08:51 +05:30
end
2023-03-17 16:20:25 +05:30
2023-07-09 08:55:56 +05:30
private_class_method def self.derive_schema_from_index(index_name)
index = Gitlab::Database::PostgresIndex.find_by(name: index_name,
schema: ApplicationRecord.connection.current_schema)
return unless index
table_schema(index.tablename)
end
2023-03-17 16:20:25 +05:30
private_class_method def self.build_dictionary(path_globs)
Dir.glob(path_globs).each_with_object({}) do |file_path, dic|
data = YAML.load_file(file_path)
key_name = data['table_name'] || data['view_name']
2023-05-27 22:25:52 +05:30
# rubocop:disable Gitlab/DocUrl
if data['gitlab_schema'].nil?
raise(
UnknownSchemaError,
2023-07-09 08:55:56 +05:30
"#{file_path} must specify a valid gitlab_schema for #{key_name}. " \
2023-05-27 22:25:52 +05:30
"See https://docs.gitlab.com/ee/development/database/database_dictionary.html"
)
end
# rubocop:enable Gitlab/DocUrl
2023-03-17 16:20:25 +05:30
dic[key_name] = data['gitlab_schema'].to_sym
end
end
2021-12-11 22:18:48 +05:30
end
end
end
2022-08-13 15:12:31 +05:30
Gitlab::Database::GitlabSchema.prepend_mod