123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- require 'rails_helper'
- RSpec.describe Service::Ticket::Create, current_user_id: -> { user.id } do
- subject(:service) { described_class.new(current_user: user) }
- let(:user) { create(:agent, groups: [group]) }
- let(:group) { create(:group) }
- let(:customer) { create(:customer) }
- describe '#execute' do
- let(:sample_title) { Faker::Lorem.sentence }
- let(:ticket_data) do
- {
- title: sample_title,
- group: group,
- customer: customer
- }
- end
- it 'creates a ticket with given metadata' do
- ticket = service.execute(ticket_data:)
- expect(ticket)
- .to have_attributes(
- title: sample_title,
- group: group,
- customer: customer
- )
- end
- it 'creates a ticket with customer email address' do
- test_email = Faker::Internet.unique.email
- ticket_data[:customer] = test_email
- ticket = service.execute(ticket_data:)
- expect(ticket.customer).to have_attributes(
- email: test_email,
- role_ids: Role.signup_role_ids
- )
- end
- it 'fails to create ticket without access' do
- allow_any_instance_of(TicketPolicy)
- .to receive(:create?).and_return(false)
- expect { service.execute(ticket_data:) }
- .to raise_error(Pundit::NotAuthorizedError)
- end
- it 'adds article when present' do
- sample_body = Faker::Lorem.sentence
- ticket_data[:article] = {
- body: sample_body
- }
- ticket = service.execute(ticket_data:)
- expect(ticket.articles.first)
- .to have_attributes(
- body: sample_body
- )
- end
- context 'when email article should be created, but to is not present' do
- let(:customer) { create(:customer, email: '') }
- let(:ticket_data) do
- {
- title: sample_title,
- group: group,
- customer: customer,
- article: {
- body: Faker::Lorem.sentence,
- type: 'email',
- to: nil,
- }
- }
- end
- it 'raises an error' do
- expect { service.execute(ticket_data:) }.to raise_error(Exceptions::InvalidAttribute, 'Sending an email without a valid recipient is not possible.')
- end
- end
- context 'when email article should be created, but to is not a valid email' do
- let(:customer) { create(:customer) }
- let(:ticket_data) do
- {
- title: sample_title,
- group: group,
- customer: customer,
- article: {
- body: Faker::Lorem.sentence,
- type: 'email',
- to: 'invalid-email',
- }
- }
- end
- it 'raises an error' do
- expect { service.execute(ticket_data:) }.to raise_error(Exceptions::InvalidAttribute, 'Sending an email without a valid recipient is not possible.')
- end
- end
- it 'adds tags when present' do
- sample_tags = [Faker::Lorem.word]
- ticket_data[:tags] = sample_tags
- ticket = service.execute(ticket_data:)
- expect(ticket.tag_list)
- .to eq sample_tags
- end
- context 'when adding links' do
- let!(:other_ticket) { create(:ticket, customer: customer) }
- let(:links) do
- [
- { link_object: other_ticket, link_type: 'child' },
- { link_object: other_ticket, link_type: 'normal' },
- ]
- end
- it 'adds links correctly' do
- ticket_data[:links] = links
- ticket = service.execute(ticket_data:)
- expect(Link.list(link_object: 'Ticket', link_object_value: ticket.id)).to contain_exactly(
- { 'link_object' => 'Ticket', 'link_object_value' => other_ticket.id, 'link_type' => 'parent' },
- { 'link_object' => 'Ticket', 'link_object_value' => other_ticket.id, 'link_type' => 'normal' },
- )
- end
- end
- context 'when tag creation is disabled' do
- before do
- Setting.set('tag_new', false)
- end
- it 'does not adds tags when present' do
- sample_tags = [Faker::Lorem.word]
- ticket_data[:tags] = sample_tags
- ticket = service.execute(ticket_data:)
- expect(ticket.tag_list).to be_empty
- end
- end
- describe 'shared draft handling' do
- let(:shared_draft) { create(:ticket_shared_draft_start, group:) }
- before { ticket_data[:shared_draft] = shared_draft }
- it 'destroys given shared draft' do
- service.execute(ticket_data:)
- expect(Ticket::SharedDraftStart).not_to exist(shared_draft.id)
- end
- it 'raises error if shared drafts are disabled on that group' do
- group.update! shared_drafts: false
- expect { service.execute(ticket_data:) }
- .to raise_error(Exceptions::UnprocessableEntity)
- end
- it 'raises error if shared draft group does not match ticket group' do
- shared_draft.update! group: create(:group)
- group.update! shared_drafts: false
- expect { service.execute(ticket_data:) }
- .to raise_error(Exceptions::UnprocessableEntity)
- end
- end
- describe 'issue trackers handling' do
- let(:gitlab_link) { 'https://git.example.com/issue/123' }
- let(:github_link) { 'https://github.com/issue/123' }
- before do
- ticket_data[:external_references] = {
- gitlab: [gitlab_link],
- github: [github_link],
- idoit: [42],
- }
- end
- context 'when none enabled' do
- it 'adds no data' do
- ticket = service.execute(ticket_data:)
- expect(ticket.preferences).to be_blank
- end
- end
- context 'when gitlab is enabled' do
- before do
- Setting.set('gitlab_integration', true)
- end
- it 'adds gitlab links' do
- ticket = service.execute(ticket_data:)
- expect(ticket.preferences).to eq({ 'gitlab' => { 'issue_links' => [gitlab_link] } })
- end
- end
- context 'when github is enabled' do
- before do
- Setting.set('github_integration', true)
- end
- it 'adds github links' do
- ticket = service.execute(ticket_data:)
- expect(ticket.preferences).to eq({ 'github' => { 'issue_links' => [github_link] } })
- end
- end
- context 'when idoit is enabled' do
- before do
- Setting.set('idoit_integration', true)
- end
- it 'adds github links' do
- ticket = service.execute(ticket_data:)
- expect(ticket.preferences).to eq({ 'idoit' => { 'object_ids' => [42] } })
- end
- end
- end
- end
- end
|