gitlab-ce/spec/services/work_items/create_service_spec.rb

229 lines
7.0 KiB
Ruby

# frozen_string_literal: true
require 'spec_helper'
RSpec.describe WorkItems::CreateService, feature_category: :team_planning do
include AfterNextHelpers
before_all do
# Ensure support bot user is created so creation doesn't count towards query limit
# and we don't try to obtain an exclusive lease within a transaction.
# See https://gitlab.com/gitlab-org/gitlab/-/issues/509629
Users::Internal.support_bot_id
end
RSpec.shared_examples 'creates work item in container' do |container_type|
include_context 'with container for work items service', container_type
describe '#execute' do
shared_examples 'fails creating work item and returns errors' do
it 'does not create new work item if parent can not be set' do
expect { service_result }.not_to change { WorkItem.count }
expect(service_result[:status]).to be(:error)
expect(service_result[:message]).to match(error_message)
end
end
subject(:service_result) { service.execute }
context 'when user is not allowed to create a work item in the container' do
let(:current_user) { user_with_no_access }
it { is_expected.to be_error }
it 'returns an access error' do
expect(service_result.errors).to contain_exactly('Operation not allowed')
end
end
context 'when params are valid' do
it 'created instance is a WorkItem' do
expect(Issuable::CommonSystemNotesService).to receive_message_chain(:new, :execute)
work_item = service_result[:work_item]
expect(work_item).to be_persisted
expect(work_item).to be_a(::WorkItem)
expect(work_item.title).to eq('Awesome work_item')
expect(work_item.description).to eq('please fix')
expect(work_item.work_item_type.base_type).to eq('issue')
end
it 'calls NewIssueWorker with correct arguments' do
expect(NewIssueWorker).to receive(:perform_async).with(Integer, current_user.id, 'WorkItem')
service_result
end
end
context 'when params are invalid' do
let(:opts) { { title: '' } }
it { is_expected.to be_error }
it 'returns validation errors' do
expect(service_result.errors).to contain_exactly("Title can't be blank")
expect(service_result[:work_item].persisted?).to be_falsey
end
end
context 'checking spam' do
let(:perform_spam_check) { true }
it 'checks for spam' do
expect_next_instance_of(WorkItem) do |instance|
expect(instance).to receive(:check_for_spam).with(user: current_user, action: :create)
end
service_result
end
context 'when `perform_spam_check` is set to `false`' do
let(:perform_spam_check) { false }
it 'does not check for spam' do
expect_next_instance_of(WorkItem) do |instance|
expect(instance).not_to receive(:check_for_spam)
end
service_result
end
end
end
it_behaves_like 'work item widgetable service' do
let(:widget_params) do
{
hierarchy_widget: { parent: parent }
}
end
let(:service) do
described_class.new(
container: container,
current_user: current_user,
params: opts,
widget_params: widget_params
)
end
let(:service_execute) { service.execute }
let(:supported_widgets) do
[
{
klass: WorkItems::Callbacks::Hierarchy,
callback: :after_create
}
]
end
end
describe 'hierarchy widget' do
let(:widget_params) { { hierarchy_widget: { parent: parent } } }
context 'when user can admin parent link' do
let(:current_user) { guest }
context 'when parent is valid work item' do
let(:opts) do
{
title: 'Awesome work_item',
description: 'please fix',
work_item_type: WorkItems::Type.default_by_type(:task)
}
end
it 'creates new work item and sets parent reference' do
expect { service_result }.to change { WorkItem.count }.by(1).and(
change { WorkItems::ParentLink.count }.by(1)
)
expect(service_result[:status]).to be(:success)
end
end
context 'when parent type is invalid' do
let_it_be(:parent) { create(:work_item, :task, **container_args) }
it_behaves_like 'fails creating work item and returns errors' do
let(:error_message) { "it's not allowed to add this type of parent item" }
end
end
end
context 'when user cannot admin parent link' do
let(:current_user) { guest }
let_it_be(:parent) { create(:work_item, :confidential, **container_args) }
let(:opts) do
{
title: 'Awesome work_item',
description: 'please fix',
work_item_type: WorkItems::Type.default_by_type(:task)
}
end
it_behaves_like 'fails creating work item and returns errors' do
let(:error_message) { 'No matching work item found. Make sure that you are adding a valid work item ID.' }
end
end
end
it "publishes WorkItems::WorkItemCreatedEvent" do
expect { service_result }
.to change { WorkItem.count }.by(1)
.and publish_event(WorkItems::WorkItemCreatedEvent).with(
id: kind_of(Numeric),
namespace_id: kind_of(Numeric)
)
end
end
end
it_behaves_like 'creates work item in container', :project
it_behaves_like 'creates work item in container', :project_namespace
context 'when applying quick actions' do
let(:service) do
described_class.new(
container: project,
current_user: user,
params: params,
widget_params: widget_params
)
end
let(:issuable) { service.execute[:work_item] }
it_behaves_like 'issuable record that supports quick actions'
it_behaves_like 'issuable record that supports quick actions', with_widgets: true
context 'when applying work item quick actions using legacy params' do
let_it_be(:project) { create(:project) }
let_it_be(:user) { project.creator }
let_it_be(:task) { create(:work_item, :task, project: project) }
let(:params) do
{
title: 'Awesome work_item',
description: "please fix\n/add_child #{task.to_reference}"
}
end
let(:widget_params) { {} }
it 'applies the quick action' do
service_result = service.execute
expect(service_result[:status]).to be(:success)
expect(service_result[:work_item].work_item_children).to contain_exactly(
task
)
end
end
end
end