Commit 62680390 authored by Philip Cunningham's avatar Philip Cunningham

Associate successful DAST validations with sites

- Amends service to associate with existing validations
- Updates specs to be easier to understand

Changelog: changed
MR: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/70140
EE: true
parent 550065c4
...@@ -38,7 +38,7 @@ module Mutations ...@@ -38,7 +38,7 @@ module Mutations
dast_site_token = dast_site_token_id.find dast_site_token = dast_site_token_id.find
response = ::DastSiteValidations::CreateService.new( response = ::DastSiteValidations::FindOrCreateService.new(
container: project, container: project,
current_user: current_user, current_user: current_user,
params: { params: {
......
# frozen_string_literal: true # frozen_string_literal: true
module DastSiteValidations module DastSiteValidations
class CreateService < BaseContainerService class FindOrCreateService < BaseContainerService
def execute def execute
return ServiceResponse.error(message: 'Insufficient permissions') unless allowed? return ServiceResponse.error(message: 'Insufficient permissions') unless allowed?
return ServiceResponse.success(payload: existing_validation) if existing_validation
dast_site_validation = create_validation! dast_site_validation = existing_successful_validation || create_validation!
return ServiceResponse.error(message: 'Site does not exist for profile') unless dast_site_validation.dast_site return ServiceResponse.error(message: 'Site does not exist for profile') unless dast_site_validation.dast_site
associate_dast_site!(dast_site_validation) associate_dast_site!(dast_site_validation)
return ServiceResponse.success(payload: dast_site_validation) if dast_site_validation.passed?
perform_runner_validation(dast_site_validation) perform_runner_validation(dast_site_validation)
rescue ActiveRecord::RecordInvalid => err rescue ActiveRecord::RecordInvalid => err
ServiceResponse.error(message: err.record.errors.full_messages) ServiceResponse.error(message: err.record.errors.full_messages)
...@@ -38,8 +39,8 @@ module DastSiteValidations ...@@ -38,8 +39,8 @@ module DastSiteValidations
@validation_strategy ||= params.fetch(:validation_strategy) @validation_strategy ||= params.fetch(:validation_strategy)
end end
def existing_validation def existing_successful_validation
@existing_validation ||= find_latest_successful_dast_site_validation @existing_successful_validation ||= find_latest_successful_dast_site_validation
end end
def url_base def url_base
......
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
require 'spec_helper' require 'spec_helper'
RSpec.describe DastSiteValidations::CreateService do RSpec.describe DastSiteValidations::FindOrCreateService do
let_it_be(:project) { create(:project, :repository) } let_it_be(:project) { create(:project, :repository) }
let_it_be(:developer) { create(:user, developer_projects: [project]) } let_it_be(:developer) { create(:user, developer_projects: [project]) }
let_it_be(:dast_site) { create(:dast_site, project: project) } let_it_be(:dast_site) { create(:dast_site, project: project) }
...@@ -12,98 +12,96 @@ RSpec.describe DastSiteValidations::CreateService do ...@@ -12,98 +12,96 @@ RSpec.describe DastSiteValidations::CreateService do
subject { described_class.new(container: project, current_user: developer, params: params).execute } subject { described_class.new(container: project, current_user: developer, params: params).execute }
shared_examples 'the licensed feature is not available' do describe 'execute', :clean_gitlab_redis_shared_state do
it 'communicates failure' do context 'when the licensed feature is available' do
stub_licensed_features(security_on_demand_scans: false) it 'communicates failure' do
stub_licensed_features(security_on_demand_scans: false)
aggregate_failures do aggregate_failures do
expect(subject.status).to eq(:error) expect(subject.status).to eq(:error)
expect(subject.message).to eq('Insufficient permissions') expect(subject.message).to eq('Insufficient permissions')
end
end end
end end
end
shared_examples 'the licensed feature is available' do context 'when the licensed feature is available' do
before do before do
stub_licensed_features(security_on_demand_scans: true) stub_licensed_features(security_on_demand_scans: true)
end end
it 'communicates success' do it 'communicates success' do
expect(subject.status).to eq(:success) expect(subject.status).to eq(:success)
end end
it 'creates a new record in the database' do it 'creates a new record in the database' do
expect { subject }.to change { DastSiteValidation.count }.by(1) expect { subject }.to change { DastSiteValidation.count }.by(1)
end end
it 'associates the dast_site_validation with the dast_site' do it 'associates the dast_site_validation with the dast_site' do
expect(subject.payload).to eq(dast_site.reload.dast_site_validation) expect(subject.payload).to eq(dast_site.reload.dast_site_validation)
end end
context 'when a param is missing' do it 'attempts to validate' do
let(:params) { { dast_site_token: dast_site_token, validation_strategy: :text_file } } expected_args = { project: project, current_user: developer, params: { dast_site_validation: instance_of(DastSiteValidation) } }
it 'communicates failure' do expect(AppSec::Dast::SiteValidations::RunnerService).to receive(:new).with(expected_args).and_call_original
aggregate_failures do
expect(subject.status).to eq(:error) subject
expect(subject.message).to eq('Key not found: :url_path')
end
end end
end
context 'when the dast_site_token.project and container do not match' do context 'when a param is missing' do
let_it_be(:dast_site_token) { create(:dast_site_token, project: create(:project), url: dast_site.url) } let(:params) { { dast_site_token: dast_site_token, validation_strategy: :text_file } }
it 'communicates failure' do it 'communicates failure' do
aggregate_failures do aggregate_failures do
expect(subject.status).to eq(:error) expect(subject.status).to eq(:error)
expect(subject.message).to eq('Insufficient permissions') expect(subject.message).to eq('Key not found: :url_path')
end
end end
end end
end
context 'when the dast_site_token does not have a related dast_site via its url' do context 'when the dast_site_token.project and container do not match' do
let_it_be(:dast_site_token) { create(:dast_site_token, project: project, url: generate(:url)) } let_it_be(:dast_site_token) { create(:dast_site_token, project: create(:project), url: dast_site.url) }
it 'communicates failure' do it 'communicates failure' do
aggregate_failures do aggregate_failures do
expect(subject.status).to eq(:error) expect(subject.status).to eq(:error)
expect(subject.message).to eq('Site does not exist for profile') expect(subject.message).to eq('Insufficient permissions')
end
end end
end end
end
end
shared_examples 'a dast_site_validation already exists' do
let!(:dast_site_validation) { create(:dast_site_validation, dast_site_token: dast_site_token, state: :passed) }
it 'returns the existing successful dast_site_validation' do context 'when the dast_site_token does not have a related dast_site via its url' do
expect(subject.payload).to eq(dast_site_validation) let_it_be(:dast_site_token) { create(:dast_site_token, project: project, url: generate(:url)) }
end
it 'does not create a new record in the database' do
expect { subject }.not_to change { DastSiteValidation.count }
end
end
describe 'execute', :clean_gitlab_redis_shared_state do it 'communicates failure' do
it_behaves_like 'the licensed feature is not available' aggregate_failures do
expect(subject.status).to eq(:error)
expect(subject.message).to eq('Site does not exist for profile')
end
end
end
it_behaves_like 'the licensed feature is available' do context 'when the site has already passed validation' do
it 'attempts to validate' do let_it_be(:dast_site_validation) { create(:dast_site_validation, dast_site_token: dast_site_token, state: :passed) }
expected_args = { project: project, current_user: developer, params: { dast_site_validation: instance_of(DastSiteValidation) } }
expect(AppSec::Dast::SiteValidations::RunnerService).to receive(:new).with(expected_args).and_call_original it 'returns the existing dast_site_validation' do
expect(subject.payload).to eq(dast_site_validation)
end
subject it 'does not create a new record in the database' do
end expect { subject }.not_to change { DastSiteValidation.count }
end
it_behaves_like 'a dast_site_validation already exists' do
it 'does not attempt to re-validate' do it 'does not attempt to re-validate' do
expect(AppSec::Dast::SiteValidations::RunnerService).not_to receive(:new) expect(AppSec::Dast::SiteValidations::RunnerService).not_to receive(:new)
subject subject
end end
it 'associates the dast_site_validation with the dast_site' do
expect(subject.payload).to eq(dast_site.reload.dast_site_validation)
end
end end
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