123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- require 'rails_helper'
- RSpec.describe 'Ticket views', authenticated_as: :authenticate, type: :system do
- def authenticate
- true
- end
- context 'when calling without session' do
- describe 'redirect to' do
- it 'login screen after certain overview was called', authenticated_as: false do
- visit '#ticket/view/all_open'
- expect(page).to have_css('#login')
- end
- it 'login screen after not overview was called', authenticated_as: false do
- visit '#ticket/view'
- expect(page).to have_css('#login')
- end
- end
- end
- context 'macros' do
- let(:group1) { create(:group) }
- let(:group2) { create(:group) }
- let(:macro_without_group) { create(:macro) }
- let(:macro_note) { create(:macro, perform: { 'article.note'=>{ 'body' => 'macro body', 'internal' => 'true', 'subject' => 'macro note' } }) }
- let(:macro_group1) { create(:macro, groups: [group1]) }
- let(:macro_group2) { create(:macro, groups: [group2]) }
- let(:ticket1) { create(:ticket, group: group1) }
- let(:ticket2) { create(:ticket, group: group2) }
- describe 'group-dependent macros' do
- let(:agent) { create(:agent, groups: Group.all) }
- def authenticate
- ticket1 && ticket2
- macro_without_group && macro_group1 && macro_group2
- agent
- end
- it 'shows only non-group macro when ticket does not match any group macros' do
- visit '#ticket/view/all_open'
- within(:active_content) do
- display_macro_batches Ticket.first
- expect(page).to have_selector(:macro_batch, macro_without_group.id)
- .and(have_no_selector(:macro_batch, macro_group1.id))
- .and(have_no_selector(:macro_batch, macro_group2.id))
- end
- end
- it 'shows non-group and matching group macros for matching ticket' do
- visit '#ticket/view/all_open'
- within(:active_content) do
- display_macro_batches ticket1
- expect(page).to have_selector(:macro_batch, macro_without_group.id)
- .and(have_selector(:macro_batch, macro_group1.id))
- .and(have_no_selector(:macro_batch, macro_group2.id))
- end
- end
- end
- describe 'macro article creation' do
- def authenticate
- macro_note
- true
- end
- it 'can use macro to create article' do
- visit '#ticket/view/all_open'
- within(:active_content) do
- display_macro_batches Ticket.first
- move_mouse_to find(:macro_batch, macro_note.id)
- release_mouse
- expect do
- wait.until { Ticket.first.articles.last.subject == 'macro note' }
- end.not_to raise_error
- end
- end
- end
- context 'when saving is blocked by one of selected tickets' do
- let(:ticket) { Ticket.first }
- let(:core_workflow_action) { { 'ticket.priority_id': { operator: 'remove_option', remove_option: '3' } } }
- let(:core_workflow) { create(:core_workflow, :active_and_screen, :perform_action) }
- let(:macro_perform) do
- {
- 'ticket.priority_id': { pre_condition: 'specific', value: 3.to_s }
- }
- end
- let(:macro_priority) { create(:macro, perform: macro_perform) }
- def authenticate
- core_workflow && macro_priority && ticket1
- true
- end
- it 'shows modal with blocking ticket title' do
- visit '#ticket/view/all_open'
- within(:active_content) do
- display_macro_batches ticket
- move_mouse_to find(:macro_batch, macro_priority.id)
- release_mouse
- in_modal do
- expect(page).to have_text(ticket.title)
- end
- end
- end
- end
- describe 'when agent cannot change some of the tickets' do
- let(:agent) { create(:agent) }
- def authenticate
- macro_without_group
- agent
- .tap { |user| user.user_groups.create! group: ticket1.group, access: 'full' }
- .tap { |user| user.user_groups.create! group: ticket2.group, access: 'overview' }
- end
- before do
- visit '#ticket/view/all_open'
- end
- it 'show macros if agent cannot change selected tickets' do
- display_macro_batches ticket1
- within(:active_content) do
- expect(page).to have_no_text(%r{No macros available}i)
- .and(have_selector(:macro_batch, macro_without_group.id))
- end
- end
- it 'show no macros if agent cannot change selected tickets' do
- display_macro_batches ticket2
- within(:active_content) do
- expect(page).to have_text(%r{No macros available}i)
- .and(have_text(%r{no change permission}i))
- .and(have_no_selector(:macro_batch, macro_without_group.id))
- end
- end
- end
- describe 'when user is agent-customer' do
- let(:agent_customer) { create(:agent_and_customer) }
- def authenticate
- ticket1.update!(customer: agent_customer)
- ticket2
- macro_without_group && macro_group1 && macro_group2
- agent_customer
- .tap { |user| user.groups << ticket2.group }
- end
- before do
- visit '#ticket/view/all_open'
- end
- it 'show no macros if the ticket is customer-like' do
- display_macro_batches ticket1
- within :active_content do
- expect(page).to have_text(%r{No macros available}i)
- .and(have_text(%r{no change permission}i))
- .and(have_no_selector(:macro_batch, macro_without_group.id))
- .and(have_no_selector(:macro_batch, macro_group1.id))
- .and(have_no_selector(:macro_batch, macro_group2.id))
- end
- end
- it 'show macros if tickets are only agent-like' do
- display_macro_batches ticket2
- within :active_content do
- expect(page).to have_no_text(%r{No macros available}i)
- .and(have_selector(:macro_batch, macro_without_group.id))
- .and(have_no_selector(:macro_batch, macro_group1.id))
- .and(have_selector(:macro_batch, macro_group2.id))
- end
- end
- end
- describe 'when user is customer' do
- let(:customer) { create(:customer) }
- def authenticate
- ticket1.update!(customer: customer)
- customer
- end
- before do
- visit '#ticket/view/my_tickets'
- end
- it 'shows no overlay' do
- display_macro_batches ticket1
- within :active_content do
- expect(page).to have_no_selector('.batch-overlay-backdrop')
- end
- end
- end
- context 'with macro batch overlay' do
- shared_examples "adding 'small' class to macro element" do
- it 'adds a "small" class to the macro element' do
- within(:active_content) do
- display_macro_batches Ticket.first
- expect(page).to have_css('.batch-overlay-macro-entry.small')
- end
- end
- end
- shared_examples "not adding 'small' class to macro element" do
- it 'does not add a "small" class to the macro element' do
- within(:active_content) do
- display_macro_batches Ticket.first
- expect(page).to have_no_selector('.batch-overlay-macro-entry.small')
- end
- end
- end
- shared_examples 'showing all macros' do
- it 'shows all macros' do
- within(:active_content) do
- display_macro_batches Ticket.first
- expect(page).to have_css('.batch-overlay-macro-entry', count: all)
- end
- end
- end
- shared_examples 'showing some macros' do |count|
- it 'shows all macros' do
- within(:active_content) do
- display_macro_batches Ticket.first
- expect(page).to have_css('.batch-overlay-macro-entry', count: count)
- end
- end
- end
- def authenticate
- Macro.destroy_all && create_list(:macro, all)
- true
- end
- before do
- visit '#ticket/view/all_open'
- end
- context 'with few macros' do
- let(:all) { 15 }
- context 'when on large screen', screen_size: :desktop do
- it_behaves_like 'showing all macros'
- it_behaves_like "not adding 'small' class to macro element"
- end
- context 'when on small screen', screen_size: :tablet do
- it_behaves_like 'showing all macros'
- it_behaves_like "not adding 'small' class to macro element"
- end
- end
- context 'with many macros' do
- let(:all) { 50 }
- context 'when on large screen', screen_size: :desktop do
- it_behaves_like 'showing some macros', 32
- end
- context 'when on small screen', screen_size: :tablet do
- it_behaves_like 'showing some macros', 24
- it_behaves_like "adding 'small' class to macro element"
- end
- end
- end
- end
- context 'when performing a Bulk action' do
- context 'when creating a Note', authenticated_as: :user do
- let(:group) { create(:group) }
- let(:user) { create(:admin, groups: [group]) }
- let(:ticket1) { create(:ticket, state_name: 'open', owner: user, group: group) }
- let(:ticket2) { create(:ticket, state_name: 'open', owner: user, group: group) }
- let(:note) { Faker::Lorem.sentence }
- it 'adds note to all selected tickets' do
- ticket1 && ticket2
- visit 'ticket/view/my_assigned'
- within :active_content do
- all('.js-checkbox-field', count: 2).each(&:click)
- click '.js-confirm'
- find('.js-confirm-step textarea').fill_in with: note
- click '.js-submit'
- end
- expect do
- wait.until { [ ticket1.articles.last&.body, ticket2.articles.last&.body ] == [note, note] }
- end.not_to raise_error
- end
- end
- # https://github.com/zammad/zammad/issues/3568
- # We need a manual ticket creation to test the correct behaviour of the bulk functionality, because of some
- # leftovers after the creation in the the javascript assets store.
- context 'when performed a manual Ticket creation', authenticated_as: :agent do
- let(:customer) { create(:customer) }
- let(:group) { Group.find_by(name: 'Users') }
- let(:agent) { create(:agent, groups: [group]) }
- let!(:template) { create(:template, :dummy_data, group: group, owner: agent, customer: customer) }
- before do
- visit 'ticket/create'
- within(:active_content) do
- use_template(template)
- click('.js-submit')
- find('.ticket-article-item')
- end
- end
- it 'check that no duplicated article was created after usage of bulk action' do
- click('.menu-item[href="#ticket/view"]')
- created_ticket_id = Ticket.last.id
- within(:active_content) do
- click("tr[data-id='#{created_ticket_id}'] .js-checkbox-field")
- find('select[name="priority_id"] option[value="1"]').select_option
- click('.js-confirm')
- click('.js-submit')
- await_empty_ajax_queue
- # Check if still only one article exists on the ticket.
- click("tr[data-id='#{created_ticket_id}'] a")
- expect(page).to have_css('.ticket-article-item', count: 1)
- end
- end
- end
- context 'when saving is blocked by one of selected tickets', authenticated_as: :pre_authentication do
- let(:core_workflow) { create(:core_workflow, :active_and_screen, :perform_action) }
- let(:ticket1) { create(:ticket, group: Group.first) }
- def pre_authentication
- core_workflow && ticket1
- true
- end
- it 'shows modal with blocking ticket title' do
- visit 'ticket/view/all_open'
- within(:active_content) do
- find("tr[data-id='#{ticket1.id}']").check('bulk', allow_label_click: true)
- select '3 high', from: 'priority_id'
- click '.js-confirm'
- click '.js-submit'
- in_modal do
- expect(page).to have_text(ticket1.title)
- end
- end
- end
- end
- end
- context 'Setting "ui_table_group_by_show_count"', authenticated_as: :authenticate, db_strategy: :reset do
- let(:custom_attribute) { create(:object_manager_attribute_select, name: 'grouptest') }
- let(:tickets) do
- [
- create(:ticket, group: Group.find_by(name: 'Users')),
- create(:ticket, group: Group.find_by(name: 'Users'), grouptest: 'key_1'),
- create(:ticket, group: Group.find_by(name: 'Users'), grouptest: 'key_2'),
- create(:ticket, group: Group.find_by(name: 'Users'), grouptest: 'key_1')
- ]
- end
- def authenticate
- custom_attribute
- ObjectManager::Attribute.migration_execute
- tickets
- Overview.find_by(name: 'Open Tickets').update(group_by: custom_attribute.name)
- Setting.set('ui_table_group_by_show_count', true)
- true
- end
- it 'shows correct ticket counts' do
- visit 'ticket/view/all_open'
- within(:active_content) do
- expect(page).to have_css('.js-tableBody td b', text: '(1)')
- .and(have_css('.js-tableBody td b', text: 'value_1 (2)'))
- .and(have_css('.js-tableBody td b', text: 'value_2 (1)'))
- end
- end
- end
- context 'Customer', authenticated_as: :authenticate do
- let(:customer) { create(:customer, :with_org) }
- let(:ticket) { create(:ticket, customer: customer) }
- def authenticate
- ticket
- customer
- end
- it 'shows ticket in my tickets' do
- visit 'ticket/view/my_tickets'
- expect(page).to have_text(ticket.title)
- end
- it 'shows ticket in my organization tickets' do
- visit 'ticket/view/my_tickets'
- click_on 'My Organization Tickets'
- expect(page).to have_text(ticket.title)
- end
- end
- describe 'Grouping by custom attribute', authenticated_as: :authenticate, db_strategy: :reset do
- def authenticate
- custom_attribute
- ObjectManager::Attribute.migration_execute
- tickets
- Overview.find_by(link: 'all_unassigned').update(group_by: custom_attribute.name)
- true
- end
- context 'when sorted by custom object date' do
- let(:custom_attribute) { create(:object_manager_attribute_date, name: 'cdate') }
- let(:tickets) do
- [
- create(:ticket, group: Group.find_by(name: 'Users'), cdate: '2021-08-18'),
- create(:ticket, group: Group.find_by(name: 'Users'), cdate: '2019-01-19'),
- create(:ticket, group: Group.find_by(name: 'Users'), cdate: '2018-01-17'),
- create(:ticket, group: Group.find_by(name: 'Users'), cdate: '2018-08-19')
- ]
- end
- it 'does show the values grouped and sorted by date key value (yyy-mm-dd) instead of display value' do
- visit 'ticket/view/all_unassigned'
- headers = all('.js-tableBody td[colspan="6"]').map(&:text)
- expect(headers).to eq ['01/17/2018', '08/19/2018', '01/19/2019', '08/18/2021', '-']
- end
- end
- context 'when sorted by custom object select', authenticated_as: :authenticate, db_strategy: :reset do
- let(:custom_attribute) do
- create(:object_manager_attribute_select,
- name: 'cselect',
- data_option_options: {
- 'a' => 'Zzz a',
- 'b' => 'Yyy b',
- 'c' => 'Xxx c',
- })
- end
- let(:tickets) do
- [
- create(:ticket, group: Group.find_by(name: 'Users'), cselect: 'a'),
- create(:ticket, group: Group.find_by(name: 'Users'), cselect: 'b'),
- create(:ticket, group: Group.find_by(name: 'Users'), cselect: 'c')
- ]
- end
- it 'does show the values grouped and sorted by display value instead of key value' do
- visit 'ticket/view/all_unassigned'
- headers = all('.js-tableBody td[colspan="6"]').map(&:text)
- expect(headers).to eq ['-', 'Xxx c', 'Yyy b', 'Zzz a']
- end
- end
- end
- context 'Overview is not refreshing #5260', authenticated_as: :agent, sessions_jobs: true do
- let(:agent) { create(:agent, groups: Group.all) }
- let(:ticket) { Ticket.first }
- let(:new_title_1) { SecureRandom.uuid }
- let(:new_title_2) { SecureRandom.uuid }
- let(:new_title_3) { SecureRandom.uuid }
- before do
- visit '#ticket/view/all_open'
- ensure_websocket
- end
- it 'does refresh the ticket after the title has changed' do
- expect(page).to have_text(ticket.title)
- ticket.update(title: new_title_1)
- expect(page).to have_text(ticket.title)
- ticket.update(title: new_title_2)
- expect(page).to have_text(ticket.title)
- ticket.update(title: new_title_3)
- expect(page).to have_text(ticket.title)
- end
- end
- end
|