Commit 1a71a1ee authored by Heinrich Lee Yu's avatar Heinrich Lee Yu

Merge branch 'bw-reference-milestone-caching' into 'master'

Enable milestones to use reference filter cache

See merge request gitlab-org/gitlab!62742
parents 2085346f 543b39f2
...@@ -11,13 +11,21 @@ module Banzai ...@@ -11,13 +11,21 @@ module Banzai
def parent_records(parent, ids) def parent_records(parent, ids)
return Label.none unless parent.is_a?(Project) || parent.is_a?(Group) return Label.none unless parent.is_a?(Project) || parent.is_a?(Group)
labels = find_labels(parent) labels = find_labels(parent)
label_ids = ids.map {|y| y[:label_id]}.compact label_ids = ids.map {|y| y[:label_id]}.compact
label_names = ids.map {|y| y[:label_name]}.compact
id_relation = labels.where(id: label_ids)
label_relation = labels.where(title: label_names)
Label.from_union([id_relation, label_relation]) unless label_ids.empty?
id_relation = labels.where(id: label_ids)
end
label_names = ids.map {|y| y[:label_name]}.compact
unless label_names.empty?
label_relation = labels.where(title: label_names)
end
return Label.none if (relation = [id_relation, label_relation].compact).empty?
Label.from_union(relation)
end end
def find_object(parent_object, id) def find_object(parent_object, id)
......
...@@ -10,19 +10,55 @@ module Banzai ...@@ -10,19 +10,55 @@ module Banzai
self.reference_type = :milestone self.reference_type = :milestone
self.object_class = Milestone self.object_class = Milestone
# Links to project milestones contain the IID, but when we're handling def parent_records(parent, ids)
# 'regular' references, we need to use the global ID to disambiguate return Milestone.none unless valid_context?(parent)
# between group and project milestones.
def find_object(parent, id)
return unless valid_context?(parent)
find_milestone_with_finder(parent, id: id) milestone_iids = ids.map {|y| y[:milestone_iid]}.compact
unless milestone_iids.empty?
iid_relation = find_milestones(parent, true).where(iid: milestone_iids)
end
milestone_names = ids.map {|y| y[:milestone_name]}.compact
unless milestone_names.empty?
milestone_relation = find_milestones(parent, false).where(name: milestone_names)
end
return Milestone.none if (relation = [iid_relation, milestone_relation].compact).empty?
Milestone.from_union(relation).includes(:project, :group)
end
def find_object(parent_object, id)
key = reference_cache.records_per_parent[parent_object].keys.find do |k|
k[:milestone_iid] == id[:milestone_iid] || k[:milestone_name] == id[:milestone_name]
end
reference_cache.records_per_parent[parent_object][key] if key
end end
def find_object_from_link(parent, iid) # Transform a symbol extracted from the text to a meaningful value
return unless valid_context?(parent) #
# This method has the contract that if a string `ref` refers to a
# record `record`, then `parse_symbol(ref) == record_identifier(record)`.
#
# This contract is slightly broken here, as we only have either the milestone_iid
# or the milestone_name, but not both. But below, we have both pieces of information.
# But it's accounted for in `find_object`
def parse_symbol(symbol, match_data)
if symbol
# when parsing links, there is no `match_data[:milestone_iid]`, but `symbol`
# holds the iid
{ milestone_iid: symbol.to_i, milestone_name: nil }
else
{ milestone_iid: match_data[:milestone_iid]&.to_i, milestone_name: match_data[:milestone_name]&.tr('"', '') }
end
end
find_milestone_with_finder(parent, iid: iid) # This method has the contract that if a string `ref` refers to a
# record `record`, then `class.parse_symbol(ref) == record_identifier(record)`.
# See note in `parse_symbol` above
def record_identifier(record)
{ milestone_iid: record.iid, milestone_name: record.name }
end end
def valid_context?(parent) def valid_context?(parent)
...@@ -50,12 +86,14 @@ module Banzai ...@@ -50,12 +86,14 @@ module Banzai
return super(text, pattern) if pattern != Milestone.reference_pattern return super(text, pattern) if pattern != Milestone.reference_pattern
milestones = {} milestones = {}
unescaped_html = unescape_html_entities(text).gsub(pattern) do |match|
milestone = find_milestone($~[:project], $~[:namespace], $~[:milestone_iid], $~[:milestone_name])
if milestone unescaped_html = unescape_html_entities(text).gsub(pattern).with_index do |match, index|
milestones[milestone.id] = yield match, milestone.id, $~[:project], $~[:namespace], $~ ident = identifier($~)
"#{REFERENCE_PLACEHOLDER}#{milestone.id}" milestone = yield match, ident, $~[:project], $~[:namespace], $~
if milestone != match
milestones[index] = milestone
"#{REFERENCE_PLACEHOLDER}#{index}"
else else
match match
end end
...@@ -66,31 +104,10 @@ module Banzai ...@@ -66,31 +104,10 @@ module Banzai
escape_with_placeholders(unescaped_html, milestones) escape_with_placeholders(unescaped_html, milestones)
end end
def find_milestone(project_ref, namespace_ref, milestone_id, milestone_name) def find_milestones(parent, find_by_iid = false)
project_path = reference_cache.full_project_path(namespace_ref, project_ref) finder_params = milestone_finder_params(parent, find_by_iid)
# Returns group if project is not found by path
parent = parent_from_ref(project_path)
return unless parent MilestonesFinder.new(finder_params).execute
milestone_params = milestone_params(milestone_id, milestone_name)
find_milestone_with_finder(parent, milestone_params)
end
def milestone_params(iid, name)
if name
{ name: name.tr('"', '') }
else
{ iid: iid.to_i }
end
end
def find_milestone_with_finder(parent, params)
finder_params = milestone_finder_params(parent, params[:iid].present?)
MilestonesFinder.new(finder_params).find_by(params)
end end
def milestone_finder_params(parent, find_by_iid) def milestone_finder_params(parent, find_by_iid)
...@@ -131,6 +148,14 @@ module Banzai ...@@ -131,6 +148,14 @@ module Banzai
def object_link_title(object, matches) def object_link_title(object, matches)
nil nil
end end
def parent
project || group
end
def requires_unescaping?
true
end
end end
end end
end end
......
...@@ -327,6 +327,7 @@ RSpec.describe Banzai::Filter::References::MilestoneReferenceFilter do ...@@ -327,6 +327,7 @@ RSpec.describe Banzai::Filter::References::MilestoneReferenceFilter do
it_behaves_like 'String-based single-word references' it_behaves_like 'String-based single-word references'
it_behaves_like 'String-based multi-word references in quotes' it_behaves_like 'String-based multi-word references in quotes'
it_behaves_like 'referencing a milestone in a link href' it_behaves_like 'referencing a milestone in a link href'
it_behaves_like 'linking to a milestone as the entire link'
it_behaves_like 'cross-project / cross-namespace complete reference' it_behaves_like 'cross-project / cross-namespace complete reference'
it_behaves_like 'cross-project / same-namespace complete reference' it_behaves_like 'cross-project / same-namespace complete reference'
it_behaves_like 'cross project shorthand reference' it_behaves_like 'cross project shorthand reference'
...@@ -460,4 +461,76 @@ RSpec.describe Banzai::Filter::References::MilestoneReferenceFilter do ...@@ -460,4 +461,76 @@ RSpec.describe Banzai::Filter::References::MilestoneReferenceFilter do
include_context 'group milestones' include_context 'group milestones'
end end
end end
context 'checking N+1' do
let_it_be(:group) { create(:group) }
let_it_be(:group2) { create(:group) }
let_it_be(:project) { create(:project, :public, namespace: group) }
let_it_be(:project2) { create(:project, :public, namespace: group2) }
let_it_be(:project3) { create(:project, :public) }
let_it_be(:project_milestone) { create(:milestone, project: project) }
let_it_be(:project_milestone2) { create(:milestone, project: project) }
let_it_be(:project2_milestone) { create(:milestone, project: project2) }
let_it_be(:group2_milestone) { create(:milestone, group: group2) }
let_it_be(:project_reference) { "#{project_milestone.to_reference}" }
let_it_be(:project_reference2) { "#{project_milestone2.to_reference}" }
let_it_be(:project2_reference) { "#{project2_milestone.to_reference(full: true)}" }
let_it_be(:group2_reference) { "#{project2.full_path}%\"#{group2_milestone.name}\"" }
it 'does not have N+1 per multiple references per project', :use_sql_query_cache do
markdown = "#{project_reference}"
control_count = 4
expect do
reference_filter(markdown)
end.not_to exceed_all_query_limit(control_count)
markdown = "#{project_reference} %qwert %werty %ertyu %rtyui #{project_reference2}"
expect do
reference_filter(markdown)
end.not_to exceed_all_query_limit(control_count)
end
it 'has N+1 for multiple unique project/group references', :use_sql_query_cache do
markdown = "#{project_reference}"
control_count = 4
expect do
reference_filter(markdown, project: project)
end.not_to exceed_all_query_limit(control_count)
# Since we're not batching milestone queries across projects/groups,
# queries increase when a new project/group is added.
# TODO: https://gitlab.com/gitlab-org/gitlab/-/issues/330359
markdown = "#{project_reference} #{group2_reference}"
control_count += 5
expect do
reference_filter(markdown)
end.not_to exceed_all_query_limit(control_count)
# third reference to already queried project/namespace, nothing extra (no N+1 here)
markdown = "#{project_reference} #{group2_reference} #{project_reference2}"
expect do
reference_filter(markdown)
end.not_to exceed_all_query_limit(control_count)
# last reference needs additional queries
markdown = "#{project_reference} #{group2_reference} #{project2_reference} #{project3.full_path}%test_milestone"
control_count += 6
expect do
reference_filter(markdown)
end.not_to exceed_all_query_limit(control_count)
# Use an iid instead of title reference
markdown = "#{project_reference} #{group2_reference} #{project2.full_path}%#{project2_milestone.iid} #{project3.full_path}%test_milestone"
expect do
reference_filter(markdown)
end.not_to exceed_all_query_limit(control_count)
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