Skip to content
代码片段 群组 项目
提交 56c71bf8 编辑于 作者: Grzegorz Bizon's avatar Grzegorz Bizon
浏览文件

Merge branch 'morefice/introduce-ci-queryanalyzers' into 'master'

Introduce Ci::PartitioningAnalyzer

See merge request gitlab-org/gitlab!97113
No related branches found
No related tags found
无相关合并请求
---
name: ci_partitioning_analyze_queries
introduced_by_url: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/97113
rollout_issue_url: https://gitlab.com/gitlab-org/gitlab/-/issues/372840
milestone: '15.4'
type: ops
group: group::pipeline execution
default_enabled: false
# frozen_string_literal: true
module Gitlab
module Database
module QueryAnalyzers
module Ci
# The purpose of this analyzer is to detect queries not going through a partitioning routing table
class PartitioningAnalyzer < Database::QueryAnalyzers::Base
RoutingTableNotUsedError = Class.new(QueryAnalyzerError)
ENABLED_TABLES = %w[
ci_builds_metadata
].freeze
class << self
def enabled?
::Feature::FlipperFeature.table_exists? &&
::Feature.enabled?(:ci_partitioning_analyze_queries, type: :ops)
end
def analyze(parsed)
analyze_legacy_tables_usage(parsed)
end
private
def analyze_legacy_tables_usage(parsed)
detected = ENABLED_TABLES & (parsed.pg.dml_tables + parsed.pg.select_tables)
return if detected.none?
::Gitlab::ErrorTracking.track_and_raise_for_dev_exception(
RoutingTableNotUsedError.new("Detected non-partitioned table use #{detected.inspect}: #{parsed.sql}")
)
end
end
end
end
end
end
end
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe Gitlab::Database::QueryAnalyzers::Ci::PartitioningAnalyzer, query_analyzers: false do
let(:analyzer) { described_class }
before do
allow(Gitlab::Database::QueryAnalyzer.instance).to receive(:all_analyzers).and_return([analyzer])
end
context 'when ci_partitioning_analyze_queries is disabled' do
before do
stub_feature_flags(ci_partitioning_analyze_queries: false)
end
it 'does not analyze the query' do
expect(analyzer).not_to receive(:analyze)
process_sql(Ci::BuildMetadata, "SELECT 1 FROM ci_builds_metadata")
end
end
context 'when ci_partitioning_analyze_queries is enabled' do
context 'when analyzing targeted tables' do
described_class::ENABLED_TABLES.each do |enabled_table|
context 'when querying a non routing table' do
it 'tracks exception' do
expect(::Gitlab::ErrorTracking).to receive(:track_and_raise_for_dev_exception)
process_sql(Ci::ApplicationRecord, "SELECT 1 FROM #{enabled_table}")
end
it 'raises RoutingTableNotUsedError' do
expect { process_sql(Ci::ApplicationRecord, "SELECT 1 FROM #{enabled_table}") }
.to raise_error(described_class::RoutingTableNotUsedError)
end
end
end
context 'when updating a record' do
it 'raises RoutingTableNotUsedError' do
expect { process_sql(Ci::BuildMetadata, "UPDATE ci_builds_metadata SET id = 1") }
.to raise_error(described_class::RoutingTableNotUsedError)
end
end
context 'when inserting a record' do
it 'raises RoutingTableNotUsedError' do
expect { process_sql(Ci::BuildMetadata, "INSERT INTO ci_builds_metadata (id) VALUES(1)") }
.to raise_error(described_class::RoutingTableNotUsedError)
end
end
end
context 'when analyzing non targeted table' do
it 'does not raise error' do
expect { process_sql(Ci::BuildMetadata, "SELECT 1 FROM projects") }
.not_to raise_error
end
end
context 'when querying a routing table' do
it 'does not raise error' do
expect { process_sql(Ci::BuildMetadata, "SELECT 1 FROM p_ci_builds_metadata") }
.not_to raise_error
end
end
end
private
def process_sql(model, sql)
Gitlab::Database::QueryAnalyzer.instance.within do
# Skip load balancer and retrieve connection assigned to model
Gitlab::Database::QueryAnalyzer.instance.send(:process_sql, sql, model.retrieve_connection)
end
end
end
0% 加载中 .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册