# frozen_string_literal: true require 'spec_helper' RSpec.describe Integrations::DroneCi, :use_clean_rails_memory_store_caching do include ReactiveCachingHelpers subject(:integration) { described_class.new } it_behaves_like Integrations::ResetSecretFields do let(:integration) { subject } end describe 'validations' do context 'active' do before do subject.active = true end it { is_expected.to validate_presence_of(:token) } it { is_expected.to validate_presence_of(:drone_url) } it_behaves_like 'issue tracker integration URL attribute', :drone_url end context 'inactive' do before do subject.active = false end it { is_expected.not_to validate_presence_of(:token) } it { is_expected.not_to validate_presence_of(:drone_url) } end end shared_context :drone_ci_integration do subject(:drone) do described_class.new( project: project, active: true, drone_url: drone_url, token: token ) end let(:project) { create(:project, :repository, name: 'project') } let(:path) { project.full_path } let(:drone_url) { 'http://drone.example.com' } let(:sha) { '2ab7834c' } let(:branch) { 'dev' } let(:token) { 'secret' } let(:iid) { rand(1..9999) } # URLs let(:build_page) { "#{drone_url}/gitlab/#{path}/redirect/commits/#{sha}?branch=#{branch}" } let(:commit_status_path) { "#{drone_url}/gitlab/#{path}/commits/#{sha}?branch=#{branch}&access_token=#{token}" } def stub_request(status: 200, body: nil) body ||= %q({"status":"success"}) WebMock.stub_request(:get, commit_status_path).to_return( status: status, headers: { 'Content-Type' => 'application/json' }, body: body ) end end include_context Integrations::EnableSslVerification do describe '#enable_ssl_verification' do before do allow(integration).to receive(:new_record?).and_return(false) end it 'returns true for a known hostname' do integration.drone_url = 'https://cloud.drone.io' expect(integration.enable_ssl_verification).to be(true) end it 'returns true for new records' do allow(integration).to receive(:new_record?).and_return(true) integration.drone_url = 'http://example.com' expect(integration.enable_ssl_verification).to be(true) end it 'returns false for an unknown hostname' do integration.drone_url = 'https://example.com' expect(integration.enable_ssl_verification).to be(false) end it 'returns false for a HTTP URL' do integration.drone_url = 'http://cloud.drone.io' expect(integration.enable_ssl_verification).to be(false) end it 'returns false for an invalid URL' do integration.drone_url = 'https://example.com:foo' expect(integration.enable_ssl_verification).to be(false) end it 'returns the persisted value if present' do integration.drone_url = 'https://cloud.drone.io' integration.enable_ssl_verification = false expect(integration.enable_ssl_verification).to be(false) end end end it_behaves_like Integrations::HasWebHook do include_context :drone_ci_integration let(:integration) { drone } let(:hook_url) { "#{drone_url}/hook?owner=#{project.namespace.full_path}&name=#{project.path}&access_token={token}" } it 'does not create a hook if project is not present' do integration.project = nil integration.instance = true expect { integration.save! }.not_to change(ServiceHook, :count) end end describe "integration page/path methods" do include_context :drone_ci_integration it { expect(drone.build_page(sha, branch)).to eq(build_page) } it { expect(drone.commit_status_path(sha, branch)).to eq(commit_status_path) } end describe '#commit_status' do include_context :drone_ci_integration it 'returns the contents of the reactive cache' do stub_reactive_cache(drone, { commit_status: 'foo' }, 'sha', 'ref') expect(drone.commit_status('sha', 'ref')).to eq('foo') end end describe '#calculate_reactive_cache' do include_context :drone_ci_integration describe '#commit_status' do subject { drone.calculate_reactive_cache(sha, branch)[:commit_status] } it 'sets commit status to :error when status is 500' do stub_request(status: 500) is_expected.to eq(:error) end it 'sets commit status to :error when status is 404' do stub_request(status: 404) is_expected.to eq(:error) end Gitlab::HTTP::HTTP_ERRORS.each do |http_error| it "sets commit status to :error with a #{http_error.name} error" do WebMock.stub_request(:get, commit_status_path) .to_raise(http_error) expect(Gitlab::ErrorTracking) .to receive(:log_exception) .with(instance_of(http_error), { project_id: project.id }) is_expected.to eq(:error) end end { "killed" => :canceled, "failure" => :failed, "error" => :failed, "success" => "success" }.each do |drone_status, our_status| it "sets commit status to #{our_status.inspect} when returned status is #{drone_status.inspect}" do stub_request(body: %Q({"status":"#{drone_status}"})) is_expected.to eq(our_status) end end end end describe "execute" do include_context :drone_ci_integration let(:user) { create(:user, username: 'username') } let(:push_sample_data) do Gitlab::DataBuilder::Push.build_sample(project, user) end it 'executes the webhook' do expect(drone).to receive(:execute_web_hook!).with(push_sample_data) drone.execute(push_sample_data) end it 'does not try to execute the webhook if the integration is not in a project' do drone.project = nil drone.instance = true expect(drone).not_to receive(:execute_web_hook!) drone.execute(push_sample_data) end end end