kanbanworkflowstimelinescrumrubyroadmapproject-planningproject-managementopenprojectangularissue-trackerifcgantt-chartganttbug-trackerboardsbcf
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
252 lines
7.7 KiB
252 lines
7.7 KiB
#-- encoding: UTF-8
|
|
#-- copyright
|
|
# OpenProject is a project management system.
|
|
# Copyright (C) 2012-2015 the OpenProject Foundation (OPF)
|
|
#
|
|
# This program is free software; you can redistribute it and/or
|
|
# modify it under the terms of the GNU General Public License version 3.
|
|
#
|
|
# OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows:
|
|
# Copyright (C) 2006-2013 Jean-Philippe Lang
|
|
# Copyright (C) 2010-2013 the ChiliProject Team
|
|
#
|
|
# This program is free software; you can redistribute it and/or
|
|
# modify it under the terms of the GNU General Public License
|
|
# as published by the Free Software Foundation; either version 2
|
|
# of the License, or (at your option) any later version.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program; if not, write to the Free Software
|
|
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
#
|
|
# See doc/COPYRIGHT.rdoc for more details.
|
|
|
|
require 'spec_helper'
|
|
|
|
describe Scm::CreateManagedRepositoryService do
|
|
let(:user) { FactoryBot.build(:user) }
|
|
let(:project) { FactoryBot.build(:project) }
|
|
|
|
let(:repository) { FactoryBot.build(:repository_subversion) }
|
|
subject(:service) { Scm::CreateManagedRepositoryService.new(repository) }
|
|
|
|
let(:config) { {} }
|
|
|
|
before do
|
|
allow(OpenProject::Configuration).to receive(:[]).and_call_original
|
|
allow(OpenProject::Configuration).to receive(:[]).with('scm').and_return(config)
|
|
end
|
|
|
|
shared_examples 'does not create a filesystem repository' do
|
|
it 'does not create a filesystem repository' do
|
|
expect(repository.managed?).to be false
|
|
expect(service.call).to be false
|
|
end
|
|
end
|
|
|
|
context 'with no managed configuration' do
|
|
it_behaves_like 'does not create a filesystem repository'
|
|
end
|
|
|
|
context 'with managed repository' do
|
|
# Must not .create a managed repository, or it will call this service itself!
|
|
let(:repository) {
|
|
repo = Repository::Subversion.new(scm_type: :managed)
|
|
repo.project = project
|
|
repo
|
|
}
|
|
|
|
context 'but no managed config' do
|
|
it 'does not create a filesystem repository' do
|
|
expect(repository.managed?).to be true
|
|
expect(service.call).to be false
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'with managed local config' do
|
|
include_context 'with tmpdir'
|
|
let(:config) {
|
|
{
|
|
subversion: { manages: File.join(tmpdir, 'svn') },
|
|
git: { manages: File.join(tmpdir, 'git') }
|
|
}
|
|
}
|
|
|
|
let(:repository) {
|
|
repo = Repository::Subversion.new(scm_type: :managed)
|
|
repo.project = project
|
|
repo.configure(:managed, nil)
|
|
|
|
# Ignore default creation
|
|
allow(repo).to receive(:create_managed_repository)
|
|
repo.save!
|
|
repo
|
|
}
|
|
|
|
before do
|
|
allow_any_instance_of(Scm::CreateLocalRepositoryJob)
|
|
.to receive(:repository).and_return(repository)
|
|
allow_any_instance_of(Scm::CreateRemoteRepositoryJob)
|
|
.to receive(:repository).and_return(repository)
|
|
end
|
|
|
|
it 'creates the repository' do
|
|
expect(service.call).to be true
|
|
perform_enqueued_jobs
|
|
expect(File.directory?(repository.root_url)).to be true
|
|
end
|
|
|
|
context 'with pre-existing path on filesystem' do
|
|
before do
|
|
allow(File).to receive(:directory?).and_return(true)
|
|
end
|
|
|
|
it 'does not create the repository' do
|
|
expect(service.call).to be false
|
|
expect(service.localized_rejected_reason)
|
|
.to eq(I18n.t('repositories.errors.exists_on_filesystem'))
|
|
end
|
|
end
|
|
|
|
context 'with a permission error occurring in the Job' do
|
|
before do
|
|
allow(Scm::CreateLocalRepositoryJob)
|
|
.to receive(:new).and_raise(Errno::EACCES)
|
|
end
|
|
|
|
it 'returns the correct error' do
|
|
expect(service.call).to be false
|
|
expect(service.localized_rejected_reason)
|
|
.to eq(I18n.t('repositories.errors.path_permission_failed',
|
|
path: repository.root_url))
|
|
end
|
|
end
|
|
|
|
context 'with an OS error occurring in the Job' do
|
|
before do
|
|
allow(Scm::CreateLocalRepositoryJob)
|
|
.to receive(:new).and_raise(Errno::ENOENT)
|
|
end
|
|
|
|
it 'returns the correct error' do
|
|
expect(service.call).to be false
|
|
expect(service.localized_rejected_reason)
|
|
.to include('An error occurred while accessing the repository in the filesystem')
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'with managed remote config', webmock: true do
|
|
let(:url) { 'http://myreposerver.example.com/api/' }
|
|
let(:config) {
|
|
{
|
|
subversion: { manages: url }
|
|
}
|
|
}
|
|
|
|
let(:repository) {
|
|
repo = FactoryBot.build(:repository_subversion, scm_type: :managed)
|
|
repo.project = project
|
|
repo.configure(:managed, nil)
|
|
repo
|
|
}
|
|
|
|
it 'detects the remote config' do
|
|
expect(repository.class.managed_remote.to_s).to eq(url)
|
|
expect(repository.class).to be_manages_remote
|
|
end
|
|
|
|
context 'with a remote callback' do
|
|
let(:returned_url) { 'file:///tmp/some/url/to/repo' }
|
|
let(:root_url) { '/tmp/some/url/to/repo' }
|
|
before do
|
|
stub_request(:post, url)
|
|
.to_return(
|
|
status: 200,
|
|
body: { url: returned_url, path: root_url }.to_json
|
|
)
|
|
end
|
|
|
|
shared_examples 'calls the callback' do
|
|
before do
|
|
# Avoid setting up a second call to the remote during save
|
|
# since we only templated the repository, not created one!
|
|
expect(repository).to receive(:save).and_return(true)
|
|
end
|
|
|
|
it do
|
|
expect(Scm::CreateRemoteRepositoryJob)
|
|
.to receive(:new).and_call_original
|
|
|
|
expect(service.call).to be true
|
|
expect(repository.root_url).to eq(root_url)
|
|
expect(repository.url).to eq(returned_url)
|
|
|
|
expect(WebMock)
|
|
.to have_requested(:post, url)
|
|
.with(body: hash_including(action: 'create'))
|
|
end
|
|
end
|
|
|
|
context 'with http' do
|
|
it_behaves_like 'calls the callback'
|
|
end
|
|
|
|
context 'with https' do
|
|
let(:url) { 'https://myreposerver.example.com/api/' }
|
|
let(:config) {
|
|
{
|
|
subversion: { manages: url, insecure: insecure }
|
|
}
|
|
}
|
|
|
|
let(:instance) { Scm::CreateRemoteRepositoryJob.new }
|
|
let(:job_call) { instance.perform(repository) }
|
|
|
|
context 'with insecure option' do
|
|
let(:insecure) { true }
|
|
|
|
it_behaves_like 'calls the callback'
|
|
it 'uses the insecure option' do
|
|
job_call
|
|
expect(instance.send(:configured_verification)).to eq(OpenSSL::SSL::VERIFY_NONE)
|
|
end
|
|
end
|
|
|
|
context 'without insecure option' do
|
|
let(:insecure) { false }
|
|
|
|
it 'uses the insecure option' do
|
|
job_call
|
|
expect(instance.send(:configured_verification)).to eq(OpenSSL::SSL::VERIFY_PEER)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'with a faulty remote callback' do
|
|
before do
|
|
stub_request(:post, url)
|
|
.to_return(status: 400, body: { success: false, message: 'An error occurred' }.to_json)
|
|
end
|
|
|
|
it 'calls the callback' do
|
|
expect(Scm::CreateRemoteRepositoryJob)
|
|
.to receive(:new).and_call_original
|
|
|
|
expect(service.call).to be false
|
|
expect(service.localized_rejected_reason)
|
|
.to eq("Calling the managed remote failed with message 'An error occurred' (Code: 400)")
|
|
expect(WebMock)
|
|
.to have_requested(:post, url)
|
|
.with(body: hash_including(action: 'create'))
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|