2014-09-02 18:07:02 +05:30
|
|
|
require 'spec_helper'
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
describe Projects::UpdateService do
|
|
|
|
include ProjectForksHelper
|
2014-09-02 18:07:02 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
let(:user) { create(:user) }
|
2017-09-10 17:25:29 +05:30
|
|
|
let(:project) do
|
|
|
|
create(:project, creator: user, namespace: user.namespace)
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
describe '#execute' do
|
|
|
|
let(:gitlab_shell) { Gitlab::Shell.new }
|
|
|
|
let(:admin) { create(:admin) }
|
2014-09-02 18:07:02 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
context 'when changing visibility level' do
|
2017-08-17 22:00:37 +05:30
|
|
|
context 'when visibility_level is INTERNAL' do
|
|
|
|
it 'updates the project to internal' do
|
2018-11-18 11:00:15 +05:30
|
|
|
expect(TodosDestroyer::ProjectPrivateWorker).not_to receive(:perform_in)
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
result = update_project(project, user, visibility_level: Gitlab::VisibilityLevel::INTERNAL)
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
expect(result).to eq({ status: :success })
|
|
|
|
expect(project).to be_internal
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
context 'when visibility_level is PUBLIC' do
|
2018-03-17 18:26:18 +05:30
|
|
|
it 'updates the project to public' do
|
2018-11-18 11:00:15 +05:30
|
|
|
expect(TodosDestroyer::ProjectPrivateWorker).not_to receive(:perform_in)
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
result = update_project(project, user, visibility_level: Gitlab::VisibilityLevel::PUBLIC)
|
2018-11-18 11:00:15 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(result).to eq({ status: :success })
|
|
|
|
expect(project).to be_public
|
|
|
|
end
|
|
|
|
end
|
2014-09-02 18:07:02 +05:30
|
|
|
|
2018-11-18 11:00:15 +05:30
|
|
|
context 'when visibility_level is PRIVATE' do
|
|
|
|
before do
|
|
|
|
project.update!(visibility_level: Gitlab::VisibilityLevel::PUBLIC)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'updates the project to private' do
|
2019-01-03 12:48:30 +05:30
|
|
|
expect(TodosDestroyer::ProjectPrivateWorker).to receive(:perform_in).with(Todo::WAIT_FOR_DELETE, project.id)
|
2018-11-18 11:00:15 +05:30
|
|
|
|
|
|
|
result = update_project(project, user, visibility_level: Gitlab::VisibilityLevel::PRIVATE)
|
|
|
|
|
|
|
|
expect(result).to eq({ status: :success })
|
|
|
|
expect(project).to be_private
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
context 'when visibility levels are restricted to PUBLIC only' do
|
|
|
|
before do
|
|
|
|
stub_application_setting(restricted_visibility_levels: [Gitlab::VisibilityLevel::PUBLIC])
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
context 'when visibility_level is INTERNAL' do
|
|
|
|
it 'updates the project to internal' do
|
|
|
|
result = update_project(project, user, visibility_level: Gitlab::VisibilityLevel::INTERNAL)
|
2018-11-18 11:00:15 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
expect(result).to eq({ status: :success })
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(project).to be_internal
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when visibility_level is PUBLIC' do
|
|
|
|
it 'does not update the project to public' do
|
|
|
|
result = update_project(project, user, visibility_level: Gitlab::VisibilityLevel::PUBLIC)
|
|
|
|
|
|
|
|
expect(result).to eq({ status: :error, message: 'New visibility level not allowed!' })
|
|
|
|
expect(project).to be_private
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when updated by an admin' do
|
|
|
|
it 'updates the project to public' do
|
|
|
|
result = update_project(project, admin, visibility_level: Gitlab::VisibilityLevel::PUBLIC)
|
2018-11-18 11:00:15 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(result).to eq({ status: :success })
|
|
|
|
expect(project).to be_public
|
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
context 'when project visibility is higher than parent group' do
|
|
|
|
let(:group) { create(:group, visibility_level: Gitlab::VisibilityLevel::INTERNAL) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
project.update(namespace: group, visibility_level: group.visibility_level)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not update project visibility level' do
|
|
|
|
result = update_project(project, admin, visibility_level: Gitlab::VisibilityLevel::PUBLIC)
|
2016-01-14 18:37:52 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(result).to eq({ status: :error, message: 'Visibility level public is not allowed in a internal group.' })
|
|
|
|
expect(project.reload).to be_internal
|
|
|
|
end
|
|
|
|
end
|
2016-01-14 18:37:52 +05:30
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
describe 'when updating project that has forks' do
|
|
|
|
let(:project) { create(:project, :internal) }
|
|
|
|
let(:forked_project) { fork_project(project) }
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
it 'updates forks visibility level when parent set to more restrictive' do
|
|
|
|
opts = { visibility_level: Gitlab::VisibilityLevel::PRIVATE }
|
2016-01-14 18:37:52 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(project).to be_internal
|
|
|
|
expect(forked_project).to be_internal
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(update_project(project, admin, opts)).to eq({ status: :success })
|
2016-01-14 18:37:52 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(project).to be_private
|
|
|
|
expect(forked_project.reload).to be_private
|
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
it 'does not update forks visibility level when parent set to less restrictive' do
|
|
|
|
opts = { visibility_level: Gitlab::VisibilityLevel::PUBLIC }
|
2016-01-14 18:37:52 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(project).to be_internal
|
|
|
|
expect(forked_project).to be_internal
|
|
|
|
|
|
|
|
expect(update_project(project, admin, opts)).to eq({ status: :success })
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(project).to be_public
|
|
|
|
expect(forked_project.reload).to be_internal
|
|
|
|
end
|
2016-01-14 18:37:52 +05:30
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
context 'when updating a default branch' do
|
|
|
|
let(:project) { create(:project, :repository) }
|
2017-09-10 17:25:29 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
it 'changes a default branch' do
|
|
|
|
update_project(project, admin, default_branch: 'feature')
|
2017-09-10 17:25:29 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(Project.find(project.id).default_branch).to eq 'feature'
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not change a default branch' do
|
|
|
|
# The branch 'unexisted-branch' does not exist.
|
|
|
|
update_project(project, admin, default_branch: 'unexisted-branch')
|
|
|
|
|
|
|
|
expect(Project.find(project.id).default_branch).to eq 'master'
|
|
|
|
end
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
|
|
|
|
2018-03-27 19:54:05 +05:30
|
|
|
context 'when we update project but not enabling a wiki' do
|
|
|
|
it 'does not try to create an empty wiki' do
|
2018-11-08 19:23:39 +05:30
|
|
|
Gitlab::Shell.new.rm_directory(project.repository_storage, project.wiki.path)
|
2018-03-27 19:54:05 +05:30
|
|
|
|
|
|
|
result = update_project(project, user, { name: 'test1' })
|
|
|
|
|
|
|
|
expect(result).to eq({ status: :success })
|
|
|
|
expect(project.wiki_repository_exists?).to be false
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'handles empty project feature attributes' do
|
|
|
|
project.project_feature.update(wiki_access_level: ProjectFeature::DISABLED)
|
|
|
|
|
|
|
|
result = update_project(project, user, { name: 'test1' })
|
|
|
|
|
|
|
|
expect(result).to eq({ status: :success })
|
|
|
|
expect(project.wiki_repository_exists?).to be false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when enabling a wiki' do
|
|
|
|
it 'creates a wiki' do
|
|
|
|
project.project_feature.update(wiki_access_level: ProjectFeature::DISABLED)
|
2018-11-08 19:23:39 +05:30
|
|
|
Gitlab::Shell.new.rm_directory(project.repository_storage, project.wiki.path)
|
2018-03-27 19:54:05 +05:30
|
|
|
|
|
|
|
result = update_project(project, user, project_feature_attributes: { wiki_access_level: ProjectFeature::ENABLED })
|
|
|
|
|
|
|
|
expect(result).to eq({ status: :success })
|
|
|
|
expect(project.wiki_repository_exists?).to be true
|
|
|
|
expect(project.wiki_enabled?).to be true
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'logs an error and creates a metric when wiki can not be created' do
|
|
|
|
project.project_feature.update(wiki_access_level: ProjectFeature::DISABLED)
|
|
|
|
|
|
|
|
expect_any_instance_of(ProjectWiki).to receive(:wiki).and_raise(ProjectWiki::CouldNotCreateWikiError)
|
|
|
|
expect_any_instance_of(described_class).to receive(:log_error).with("Could not create wiki for #{project.full_name}")
|
|
|
|
expect(Gitlab::Metrics).to receive(:counter)
|
|
|
|
|
|
|
|
update_project(project, user, project_feature_attributes: { wiki_access_level: ProjectFeature::ENABLED })
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-11-18 11:00:15 +05:30
|
|
|
context 'when changing feature visibility to private' do
|
|
|
|
it 'updates the visibility correctly' do
|
|
|
|
expect(TodosDestroyer::PrivateFeaturesWorker)
|
2019-01-03 12:48:30 +05:30
|
|
|
.to receive(:perform_in).with(Todo::WAIT_FOR_DELETE, project.id)
|
2018-11-18 11:00:15 +05:30
|
|
|
|
|
|
|
result = update_project(project, user, project_feature_attributes:
|
|
|
|
{ issues_access_level: ProjectFeature::PRIVATE }
|
|
|
|
)
|
|
|
|
|
|
|
|
expect(result).to eq({ status: :success })
|
|
|
|
expect(project.project_feature.issues_access_level).to be(ProjectFeature::PRIVATE)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
context 'when updating a project that contains container images' do
|
|
|
|
before do
|
|
|
|
stub_container_registry_config(enabled: true)
|
|
|
|
stub_container_registry_tags(repository: /image/, tags: %w[rc1])
|
|
|
|
create(:container_repository, project: project, name: :image)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not allow to rename the project' do
|
|
|
|
result = update_project(project, admin, path: 'renamed')
|
|
|
|
|
|
|
|
expect(result).to include(status: :error)
|
|
|
|
expect(result[:message]).to match(/contains container registry tags/)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'allows to update other settings' do
|
|
|
|
result = update_project(project, admin, public_builds: true)
|
|
|
|
|
|
|
|
expect(result[:status]).to eq :success
|
|
|
|
expect(project.reload.public_builds).to be true
|
|
|
|
end
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
context 'when renaming a project' do
|
|
|
|
let(:repository_storage) { 'default' }
|
2018-05-09 12:01:36 +05:30
|
|
|
let(:repository_storage_path) { Gitlab.config.repositories.storages[repository_storage].legacy_disk_path }
|
2018-03-17 18:26:18 +05:30
|
|
|
|
|
|
|
context 'with legacy storage' do
|
2018-03-27 19:54:05 +05:30
|
|
|
let(:project) { create(:project, :legacy_storage, :repository, creator: user, namespace: user.namespace) }
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
before do
|
2019-03-02 22:35:43 +05:30
|
|
|
gitlab_shell.create_repository(repository_storage, "#{user.namespace.full_path}/existing", user.namespace.full_path)
|
2018-03-17 18:26:18 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
after do
|
2018-10-15 14:42:47 +05:30
|
|
|
gitlab_shell.remove_repository(repository_storage, "#{user.namespace.full_path}/existing")
|
2018-03-17 18:26:18 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not allow renaming when new path matches existing repository on disk' do
|
|
|
|
result = update_project(project, admin, path: 'existing')
|
|
|
|
|
|
|
|
expect(result).to include(status: :error)
|
|
|
|
expect(result[:message]).to match('There is already a repository with that name on disk')
|
|
|
|
expect(project).not_to be_valid
|
|
|
|
expect(project.errors.messages).to have_key(:base)
|
|
|
|
expect(project.errors.messages[:base]).to include('There is already a repository with that name on disk')
|
|
|
|
end
|
2018-11-18 11:00:15 +05:30
|
|
|
|
2018-11-20 20:47:30 +05:30
|
|
|
it 'renames the project without upgrading it' do
|
|
|
|
result = update_project(project, admin, path: 'new-path')
|
|
|
|
|
|
|
|
expect(result).not_to include(status: :error)
|
|
|
|
expect(project).to be_valid
|
|
|
|
expect(project.errors).to be_empty
|
|
|
|
expect(project.disk_path).to include('new-path')
|
|
|
|
expect(project.reload.hashed_storage?(:repository)).to be_falsey
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when hashed storage is enabled' do
|
2018-11-18 11:00:15 +05:30
|
|
|
before do
|
|
|
|
stub_application_setting(hashed_storage_enabled: true)
|
2018-11-20 20:47:30 +05:30
|
|
|
stub_feature_flags(skip_hashed_storage_upgrade: false)
|
2018-11-18 11:00:15 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'migrates project to a hashed storage instead of renaming the repo to another legacy name' do
|
|
|
|
result = update_project(project, admin, path: 'new-path')
|
|
|
|
|
|
|
|
expect(result).not_to include(status: :error)
|
|
|
|
expect(project).to be_valid
|
|
|
|
expect(project.errors).to be_empty
|
|
|
|
expect(project.reload.hashed_storage?(:repository)).to be_truthy
|
|
|
|
end
|
2018-11-20 20:47:30 +05:30
|
|
|
|
|
|
|
context 'when skip_hashed_storage_upgrade feature flag is enabled' do
|
|
|
|
before do
|
|
|
|
stub_feature_flags(skip_hashed_storage_upgrade: true)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'renames the project without upgrading it' do
|
|
|
|
result = update_project(project, admin, path: 'new-path')
|
|
|
|
|
|
|
|
expect(result).not_to include(status: :error)
|
|
|
|
expect(project).to be_valid
|
|
|
|
expect(project.errors).to be_empty
|
|
|
|
expect(project.disk_path).to include('new-path')
|
|
|
|
expect(project.reload.hashed_storage?(:repository)).to be_falsey
|
|
|
|
end
|
|
|
|
end
|
2018-11-18 11:00:15 +05:30
|
|
|
end
|
2018-03-17 18:26:18 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
context 'with hashed storage' do
|
|
|
|
let(:project) { create(:project, :repository, creator: user, namespace: user.namespace) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
stub_application_setting(hashed_storage_enabled: true)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not check if new path matches existing repository on disk' do
|
|
|
|
expect(project).not_to receive(:repository_with_same_path_already_exists?)
|
|
|
|
|
|
|
|
result = update_project(project, admin, path: 'existing')
|
2017-09-10 17:25:29 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(result).to include(status: :success)
|
|
|
|
end
|
|
|
|
end
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
context 'when passing invalid parameters' do
|
|
|
|
it 'returns an error result when record cannot be updated' do
|
|
|
|
result = update_project(project, admin, { name: 'foo&bar' })
|
2017-09-10 17:25:29 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(result).to eq({
|
|
|
|
status: :error,
|
|
|
|
message: "Name can contain only letters, digits, emojis, '_', '.', dash, space. It must start with letter, digit, emoji or '_'."
|
|
|
|
})
|
|
|
|
end
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
2018-05-09 12:01:36 +05:30
|
|
|
|
|
|
|
context 'when updating #pages_https_only', :https_pages_enabled do
|
|
|
|
subject(:call_service) do
|
|
|
|
update_project(project, admin, pages_https_only: false)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'updates the attribute' do
|
|
|
|
expect { call_service }
|
|
|
|
.to change { project.pages_https_only? }
|
|
|
|
.to(false)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'calls Projects::UpdatePagesConfigurationService' do
|
|
|
|
expect(Projects::UpdatePagesConfigurationService)
|
|
|
|
.to receive(:new)
|
|
|
|
.with(project)
|
|
|
|
.and_call_original
|
|
|
|
|
|
|
|
call_service
|
|
|
|
end
|
|
|
|
end
|
2018-12-05 23:21:45 +05:30
|
|
|
|
|
|
|
context 'when updating #pages_access_level' do
|
|
|
|
subject(:call_service) do
|
|
|
|
update_project(project, admin, project_feature_attributes: { pages_access_level: ProjectFeature::PRIVATE })
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'updates the attribute' do
|
|
|
|
expect { call_service }
|
|
|
|
.to change { project.project_feature.pages_access_level }
|
|
|
|
.to(ProjectFeature::PRIVATE)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'calls Projects::UpdatePagesConfigurationService' do
|
|
|
|
expect(Projects::UpdatePagesConfigurationService)
|
|
|
|
.to receive(:new)
|
|
|
|
.with(project)
|
|
|
|
.and_call_original
|
|
|
|
|
|
|
|
call_service
|
|
|
|
end
|
|
|
|
end
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
describe '#run_auto_devops_pipeline?' do
|
|
|
|
subject { described_class.new(project, user).run_auto_devops_pipeline? }
|
|
|
|
|
|
|
|
context 'when master contains a .gitlab-ci.yml file' do
|
|
|
|
before do
|
|
|
|
allow(project.repository).to receive(:gitlab_ci_yml).and_return("script: ['test']")
|
|
|
|
end
|
2017-09-10 17:25:29 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
it { is_expected.to eq(false) }
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
|
|
|
|
2018-11-08 19:23:39 +05:30
|
|
|
context 'when auto devops is nil' do
|
|
|
|
it { is_expected.to eq(false) }
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
context 'when auto devops is explicitly enabled' do
|
|
|
|
before do
|
|
|
|
project.create_auto_devops!(enabled: true)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to eq(true) }
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
context 'when auto devops is explicitly disabled' do
|
|
|
|
before do
|
|
|
|
project.create_auto_devops!(enabled: false)
|
|
|
|
end
|
2017-09-10 17:25:29 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
it { is_expected.to eq(false) }
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
context 'when auto devops is set to instance setting' do
|
|
|
|
before do
|
|
|
|
project.create_auto_devops!(enabled: nil)
|
|
|
|
allow(project.auto_devops).to receive(:previous_changes).and_return('enabled' => true)
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when auto devops is enabled system-wide' do
|
|
|
|
before do
|
|
|
|
stub_application_setting(auto_devops_enabled: true)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to eq(true) }
|
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
context 'when auto devops is disabled system-wide' do
|
|
|
|
before do
|
|
|
|
stub_application_setting(auto_devops_enabled: false)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to eq(false) }
|
|
|
|
end
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
2014-09-02 18:07:02 +05:30
|
|
|
def update_project(project, user, opts)
|
2017-08-17 22:00:37 +05:30
|
|
|
described_class.new(project, user, opts).execute
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|
|
|
|
end
|