Commit 3b3fcea8 authored by Philip Cunningham's avatar Philip Cunningham Committed by Heinrich Lee Yu

Reuse existing DastSiteToken if it already exists

parent 5774f6f5
...@@ -32,8 +32,8 @@ module Mutations ...@@ -32,8 +32,8 @@ module Mutations
def resolve(full_path:, target_url:) def resolve(full_path:, target_url:)
project = authorized_find!(full_path) project = authorized_find!(full_path)
response = ::DastSiteTokens::CreateService.new( response = ::AppSec::Dast::SiteTokens::FindOrCreateService.new(
container: project, project: project,
params: { target_url: target_url } params: { target_url: target_url }
).execute ).execute
......
...@@ -4,8 +4,8 @@ class DastSiteToken < ApplicationRecord ...@@ -4,8 +4,8 @@ class DastSiteToken < ApplicationRecord
belongs_to :project belongs_to :project
validates :project_id, presence: true validates :project_id, presence: true
validates :token, length: { maximum: 255 }, presence: true validates :token, length: { maximum: 255 }, presence: true, uniqueness: true
validates :url, length: { maximum: 255 }, presence: true, public_url: true validates :url, length: { maximum: 255 }, presence: true, public_url: true, uniqueness: { scope: :project_id }
def dast_site def dast_site
@dast_site ||= DastSite.find_by(project_id: project.id, url: url) @dast_site ||= DastSite.find_by(project_id: project.id, url: url)
......
# frozen_string_literal: true
module AppSec
module Dast
module SiteTokens
class FindOrCreateService < BaseProjectService
def execute
return ServiceResponse.error(message: 'Insufficient permissions') unless allowed?
existing_validation = find_dast_site_validation
return success_response(existing_validation.dast_site_token, existing_validation.state) if existing_validation
find_or_create_dast_site_token
rescue URI::InvalidURIError
error_response('Invalid target_url')
end
private
def allowed?
project.licensed_feature_available?(:security_on_demand_scans)
end
def error_response(message)
ServiceResponse.error(message: message)
end
def success_response(dast_site_token, status)
ServiceResponse.success(payload: { dast_site_token: dast_site_token, status: status })
end
def find_or_create_dast_site_token
existing_token = DastSiteToken.find_by(project: project, url: params[:target_url]) # rubocop: disable CodeReuse/ActiveRecord
return success_response(existing_token, DastSiteValidation::INITIAL_STATE) if existing_token
new_token = DastSiteToken.create(project: project, token: SecureRandom.uuid, url: params[:target_url])
return error_response(new_token.errors.full_messages) unless new_token.valid?
success_response(new_token, DastSiteValidation::INITIAL_STATE)
end
def find_dast_site_validation
url_base = DastSiteValidation.get_normalized_url_base(params[:target_url])
DastSiteValidationsFinder.new(project_id: project.id, url_base: url_base)
.execute
.first
end
end
end
end
end
# frozen_string_literal: true
module DastSiteTokens
class CreateService < BaseContainerService
def execute
return ServiceResponse.error(message: 'Insufficient permissions') unless allowed?
target_url = params[:target_url]
url_base = normalize_target_url(target_url)
dast_site_token = DastSiteToken.create!(
project: container,
token: SecureRandom.uuid,
url: target_url
)
dast_site_validation = find_dast_site_validation(url_base)
status = calculate_status(dast_site_validation)
ServiceResponse.success(
payload: { dast_site_token: dast_site_token, status: status }
)
rescue ActiveRecord::RecordInvalid => err
ServiceResponse.error(message: err.record.errors.full_messages)
rescue URI::InvalidURIError
ServiceResponse.error(message: 'Invalid target_url')
end
private
def allowed?
container.feature_available?(:security_on_demand_scans)
end
def normalize_target_url(target_url)
DastSiteValidation.get_normalized_url_base(target_url)
end
def find_dast_site_validation(url_base)
DastSiteValidationsFinder.new(project_id: container.id, url_base: url_base)
.execute
.first
end
def calculate_status(dast_site_validation)
dast_site_validation&.state || DastSiteValidation::INITIAL_STATE
end
end
end
...@@ -16,6 +16,8 @@ RSpec.describe DastSiteToken, type: :model do ...@@ -16,6 +16,8 @@ RSpec.describe DastSiteToken, type: :model do
it { is_expected.to validate_length_of(:url).is_at_most(255) } it { is_expected.to validate_length_of(:url).is_at_most(255) }
it { is_expected.to validate_presence_of(:token) } it { is_expected.to validate_presence_of(:token) }
it { is_expected.to validate_presence_of(:url) } it { is_expected.to validate_presence_of(:url) }
it { is_expected.to validate_uniqueness_of(:token) }
it { is_expected.to validate_uniqueness_of(:url).scoped_to(:project_id) }
context 'when the url is not public' do context 'when the url is not public' do
subject { build(:dast_site_token, url: 'http://127.0.0.1') } subject { build(:dast_site_token, url: 'http://127.0.0.1') }
......
...@@ -2,13 +2,13 @@ ...@@ -2,13 +2,13 @@
require 'spec_helper' require 'spec_helper'
RSpec.describe DastSiteTokens::CreateService do RSpec.describe AppSec::Dast::SiteTokens::FindOrCreateService do
let(:project) { create(:project) } let_it_be(:project) { create(:project) }
let(:target_url) { generate(:url) } let_it_be(:target_url) { generate(:url) }
subject do subject do
described_class.new( described_class.new(
container: project, project: project,
params: { target_url: target_url } params: { target_url: target_url }
).execute ).execute
end end
...@@ -18,10 +18,7 @@ RSpec.describe DastSiteTokens::CreateService do ...@@ -18,10 +18,7 @@ RSpec.describe DastSiteTokens::CreateService do
it 'communicates failure' do it 'communicates failure' do
stub_licensed_features(security_on_demand_scans: false) stub_licensed_features(security_on_demand_scans: false)
aggregate_failures do expect(subject).to have_attributes(status: :error, message: 'Insufficient permissions')
expect(subject.status).to eq(:error)
expect(subject.message).to eq('Insufficient permissions')
end
end end
end end
...@@ -30,26 +27,39 @@ RSpec.describe DastSiteTokens::CreateService do ...@@ -30,26 +27,39 @@ RSpec.describe DastSiteTokens::CreateService do
stub_licensed_features(security_on_demand_scans: true) stub_licensed_features(security_on_demand_scans: true)
end end
it 'communicates success' do it 'creates a new token' do
expect(subject.status).to eq(:success) expect { subject }.to change { DastSiteToken.count }.by(1)
end end
it 'contains a dast_site_validation' do it 'communicates success' do
expect(subject.payload[:dast_site_token]).to be_a(DastSiteToken) expect(subject).to have_attributes(status: :success, payload: { dast_site_token: instance_of(DastSiteToken), status: 'pending' })
end end
it 'contains a status' do context 'when the token already exists' do
expect(subject.payload[:status]).to eq('pending') let_it_be(:dast_site_token) { create(:dast_site_token, project: project, url: target_url) }
it 'does not create a new token' do
expect { subject }.not_to change { DastSiteToken.count }
end
it 'includes it in the payload' do
expect(subject).to have_attributes(status: :success, payload: hash_including(dast_site_token: dast_site_token))
end
context 'when an existing validation exists' do
let_it_be(:dast_site_validation) { create(:dast_site_validation, dast_site_token: dast_site_token, state: :passed) }
it 'includes its status in the payload' do
expect(subject).to have_attributes(status: :success, payload: hash_including(status: dast_site_validation.state))
end
end
end end
context 'when an invalid target_url is supplied' do context 'when an invalid target_url is supplied' do
let(:target_url) { 'http://bogus:broken' } let_it_be(:target_url) { 'http://bogus:broken' }
it 'communicates failure' do it 'communicates failure' do
aggregate_failures do expect(subject).to have_attributes(status: :error, message: 'Invalid target_url')
expect(subject.status).to eq(:error)
expect(subject.message).to eq('Invalid target_url')
end
end end
it 'does not create a dast_site_validation' do it 'does not create a dast_site_validation' do
......
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