Commit e1dad9f6 authored by Heinrich Lee Yu's avatar Heinrich Lee Yu

Fix N+1 queries when loading vulnerablity feedback

Use Vulnerabilities::OccurrencesPreloader for API requests
parent a02dcee4
---
title: Fix N+1 queries in vulnerabilities API
merge_request: 14638
author:
type: performance
...@@ -50,11 +50,15 @@ module API ...@@ -50,11 +50,15 @@ module API
get ':id/vulnerabilities' do get ':id/vulnerabilities' do
authorize! :read_project_security_dashboard, user_project authorize! :read_project_security_dashboard, user_project
vulnerability_occurrences = Kaminari.paginate_array( vulnerability_occurrences = paginate(
Kaminari.paginate_array(
vulnerability_occurrences_by(declared_params.merge(project: user_project)) vulnerability_occurrences_by(declared_params.merge(project: user_project))
) )
)
Gitlab::Vulnerabilities::OccurrencesPreloader.preload_feedback!(vulnerability_occurrences)
present paginate(vulnerability_occurrences), present vulnerability_occurrences,
with: ::Vulnerabilities::OccurrenceEntity, with: ::Vulnerabilities::OccurrenceEntity,
request: GrapeRequestProxy.new(request, current_user) request: GrapeRequestProxy.new(request, current_user)
end end
......
...@@ -9,8 +9,15 @@ module Gitlab ...@@ -9,8 +9,15 @@ module Gitlab
class OccurrencesPreloader class OccurrencesPreloader
def self.preload!(occurrences) def self.preload!(occurrences)
occurrences.all_preloaded.tap do |occurrences| occurrences.all_preloaded.tap do |occurrences|
occurrences.each(&:issue_feedback) preload_feedback!(occurrences)
occurrences.each(&:dismissal_feedback) end
end
def self.preload_feedback!(occurrences)
occurrences.each do |occurrence|
occurrence.dismissal_feedback
occurrence.issue_feedback
occurrence.merge_request_feedback
end end
end end
end end
......
...@@ -52,6 +52,14 @@ describe API::Vulnerabilities do ...@@ -52,6 +52,14 @@ describe API::Vulnerabilities do
expect(json_response.map { |v| v['report_type'] }.uniq).to match_array %w[dependency_scanning sast] expect(json_response.map { |v| v['report_type'] }.uniq).to match_array %w[dependency_scanning sast]
end end
it 'does not have N+1 queries' do
control_count = ActiveRecord::QueryRecorder.new do
get api("/projects/#{project.id}/vulnerabilities", user), params: { report_type: 'dependency_scanning' }
end.count
expect { get api("/projects/#{project.id}/vulnerabilities", user) }.not_to exceed_query_limit(control_count)
end
describe 'filtering' do describe 'filtering' do
it 'returns vulnerabilities with sast report_type' do it 'returns vulnerabilities with sast report_type' do
occurrence_count = (sast_report.occurrences.count - 1).to_s occurrence_count = (sast_report.occurrences.count - 1).to_s
......
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