Commit 2c9532e8 authored by Matthias Käppler's avatar Matthias Käppler Committed by Kerri Miller

Batch-load vulnerability findings by UUID

This prevents an N+1 where the StoreReportService
would previously query for a Finding by UUID one-by-one
only to fall back on a different query/creation when
not found.

Instead, we now pre-load findings into a Hash by UUID,
and only fall back in case we haven't already found it.
parent 23c99671
---
title: Batch-load vulnerability findings by UUID
merge_request: 55642
author:
type: performance
...@@ -33,7 +33,15 @@ module Security ...@@ -33,7 +33,15 @@ module Security
end end
def create_all_vulnerabilities! def create_all_vulnerabilities!
@report.findings.map { |finding| create_vulnerability_finding(finding)&.id }.compact.uniq # Look for existing Findings using UUID
finding_uuids = @report.findings.map(&:uuid)
vulnerability_findings_by_uuid = project.vulnerability_findings
.where(uuid: finding_uuids) # rubocop: disable CodeReuse/ActiveRecord
.to_h { |vf| [vf.uuid, vf] }
@report.findings.map do |finding|
create_vulnerability_finding(vulnerability_findings_by_uuid, finding)&.id
end.compact.uniq
end end
def mark_as_resolved_except(vulnerability_ids) def mark_as_resolved_except(vulnerability_ids)
...@@ -43,7 +51,7 @@ module Security ...@@ -43,7 +51,7 @@ module Security
.update_all(resolved_on_default_branch: true) .update_all(resolved_on_default_branch: true)
end end
def create_vulnerability_finding(finding) def create_vulnerability_finding(vulnerability_findings_by_uuid, finding)
unless finding.valid? unless finding.valid?
put_warning_for(finding) put_warning_for(finding)
return return
...@@ -51,7 +59,9 @@ module Security ...@@ -51,7 +59,9 @@ module Security
vulnerability_params = finding.to_hash.except(:compare_key, :identifiers, :location, :scanner, :scan, :links) vulnerability_params = finding.to_hash.except(:compare_key, :identifiers, :location, :scanner, :scan, :links)
entity_params = Gitlab::Json.parse(vulnerability_params&.dig(:raw_metadata)).slice('description', 'message', 'solution', 'cve', 'location') entity_params = Gitlab::Json.parse(vulnerability_params&.dig(:raw_metadata)).slice('description', 'message', 'solution', 'cve', 'location')
vulnerability_finding = create_or_find_vulnerability_finding(finding, vulnerability_params.merge(entity_params)) # Vulnerabilities::Finding (`vulnerability_occurrences`)
vulnerability_finding = vulnerability_findings_by_uuid[finding.uuid] ||
create_new_vulnerability_finding(finding, vulnerability_params.merge(entity_params))
update_vulnerability_scanner(finding) update_vulnerability_scanner(finding)
...@@ -73,7 +83,7 @@ module Security ...@@ -73,7 +83,7 @@ module Security
end end
# rubocop: disable CodeReuse/ActiveRecord # rubocop: disable CodeReuse/ActiveRecord
def create_or_find_vulnerability_finding(finding, create_params) def create_new_vulnerability_finding(finding, create_params)
find_params = { find_params = {
scanner: scanners_objects[finding.scanner.key], scanner: scanners_objects[finding.scanner.key],
primary_identifier: identifiers_objects[finding.primary_identifier.key], primary_identifier: identifiers_objects[finding.primary_identifier.key],
...@@ -81,21 +91,15 @@ module Security ...@@ -81,21 +91,15 @@ module Security
} }
begin begin
# Look for existing Findings using UUID
vulnerability_finding = project.vulnerability_findings.find_by(uuid: finding.uuid)
# If there's no Finding then we're dealing with one of two cases: # If there's no Finding then we're dealing with one of two cases:
# 1. The Finding is a new one # 1. The Finding is a new one
# 2. The Finding is already saved but has UUIDv4 # 2. The Finding is already saved but has UUIDv4
unless vulnerability_finding project.vulnerability_findings
vulnerability_finding = project.vulnerability_findings
.create_with(create_params) .create_with(create_params)
.find_or_initialize_by(find_params) .find_or_initialize_by(find_params).tap do |f|
vulnerability_finding.uuid = finding.uuid f.uuid = finding.uuid
f.save!
end end
vulnerability_finding.save!
vulnerability_finding
rescue ActiveRecord::RecordNotUnique => e rescue ActiveRecord::RecordNotUnique => e
# This might happen if we're processing another report in parallel and it finds the same Finding # This might happen if we're processing another report in parallel and it finds the same Finding
# faster. In that case we need to perform the lookup again # faster. In that case we need to perform the lookup again
......
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