2023-04-23 21:23:45 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
module Gitlab
|
|
|
|
module Database
|
|
|
|
module SchemaValidation
|
|
|
|
class StructureSql
|
2023-05-27 22:25:52 +05:30
|
|
|
DEFAULT_SCHEMA = 'public'
|
|
|
|
|
|
|
|
def initialize(structure_file_path, schema_name = DEFAULT_SCHEMA)
|
2023-04-23 21:23:45 +05:30
|
|
|
@structure_file_path = structure_file_path
|
2023-05-27 22:25:52 +05:30
|
|
|
@schema_name = schema_name
|
|
|
|
end
|
|
|
|
|
|
|
|
def index_exists?(index_name)
|
|
|
|
indexes.find { |index| index.name == index_name }.present?
|
|
|
|
end
|
|
|
|
|
|
|
|
def trigger_exists?(trigger_name)
|
|
|
|
triggers.find { |trigger| trigger.name == trigger_name }.present?
|
2023-04-23 21:23:45 +05:30
|
|
|
end
|
|
|
|
|
2023-06-20 00:43:36 +05:30
|
|
|
def fetch_table_by_name(table_name)
|
|
|
|
tables.find { |table| table.name == table_name }
|
|
|
|
end
|
|
|
|
|
|
|
|
def table_exists?(table_name)
|
|
|
|
fetch_table_by_name(table_name).present?
|
|
|
|
end
|
|
|
|
|
2023-04-23 21:23:45 +05:30
|
|
|
def indexes
|
2023-05-27 22:25:52 +05:30
|
|
|
@indexes ||= map_with_default_schema(index_statements, SchemaObjects::Index)
|
|
|
|
end
|
2023-04-23 21:23:45 +05:30
|
|
|
|
2023-05-27 22:25:52 +05:30
|
|
|
def triggers
|
|
|
|
@triggers ||= map_with_default_schema(trigger_statements, SchemaObjects::Trigger)
|
2023-04-23 21:23:45 +05:30
|
|
|
end
|
|
|
|
|
2023-06-20 00:43:36 +05:30
|
|
|
def tables
|
|
|
|
@tables ||= table_statements.map do |stmt|
|
|
|
|
table_name = stmt.relation.relname
|
2023-07-09 08:55:56 +05:30
|
|
|
partition_stmt = stmt.partspec
|
2023-06-20 00:43:36 +05:30
|
|
|
|
|
|
|
columns = stmt.table_elts.select { |n| n.node == :column_def }.map do |column|
|
2023-07-09 08:55:56 +05:30
|
|
|
adapter = Adapters::ColumnStructureSqlAdapter.new(table_name, column.column_def, partition_stmt)
|
|
|
|
SchemaObjects::Column.new(adapter)
|
2023-06-20 00:43:36 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
SchemaObjects::Table.new(table_name, columns)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-04-23 21:23:45 +05:30
|
|
|
private
|
|
|
|
|
2023-05-27 22:25:52 +05:30
|
|
|
attr_reader :structure_file_path, :schema_name
|
2023-04-23 21:23:45 +05:30
|
|
|
|
|
|
|
def index_statements
|
2023-05-27 22:25:52 +05:30
|
|
|
statements.filter_map { |s| s.stmt.index_stmt }
|
|
|
|
end
|
|
|
|
|
|
|
|
def trigger_statements
|
|
|
|
statements.filter_map { |s| s.stmt.create_trig_stmt }
|
|
|
|
end
|
|
|
|
|
2023-06-20 00:43:36 +05:30
|
|
|
def table_statements
|
|
|
|
statements.filter_map { |s| s.stmt.create_stmt }
|
|
|
|
end
|
|
|
|
|
2023-05-27 22:25:52 +05:30
|
|
|
def statements
|
|
|
|
@statements ||= parsed_structure_file.tree.stmts
|
2023-04-23 21:23:45 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def parsed_structure_file
|
|
|
|
PgQuery.parse(File.read(structure_file_path))
|
|
|
|
end
|
2023-05-27 22:25:52 +05:30
|
|
|
|
|
|
|
def map_with_default_schema(statements, validation_class)
|
|
|
|
statements.map do |statement|
|
|
|
|
statement.relation.schemaname = schema_name if statement.relation.schemaname == ''
|
|
|
|
|
|
|
|
validation_class.new(statement)
|
|
|
|
end
|
|
|
|
end
|
2023-04-23 21:23:45 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|