123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421 |
- // Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- import { getNode } from '@formkit/core'
- import { waitFor } from '@testing-library/vue'
- import { flushPromises } from '@vue/test-utils'
- import type { ExtendedRenderResult } from '#tests/support/components/index.ts'
- import { getTestRouter } from '#tests/support/components/renderComponent.ts'
- import { visitView } from '#tests/support/components/visitView.ts'
- import { mockApplicationConfig } from '#tests/support/mock-applicationConfig.ts'
- import { mockGraphQLApi } from '#tests/support/mock-graphql-api.ts'
- import { mockPermissions } from '#tests/support/mock-permissions.ts'
- import { setupView } from '#tests/support/mock-user.ts'
- import { mockUserCurrent } from '#tests/support/mock-userCurrent.ts'
- import { mockTicketOverviews } from '#tests/support/mocks/ticket-overviews.ts'
- import { nullableMock, waitUntil } from '#tests/support/utils.ts'
- import { AutocompleteSearchUserDocument } from '#shared/components/Form/fields/FieldCustomer/graphql/queries/autocompleteSearch/user.api.ts'
- import { FormUpdaterDocument } from '#shared/components/Form/graphql/queries/formUpdater.api.ts'
- import { ObjectManagerFrontendAttributesDocument } from '#shared/entities/object-attributes/graphql/queries/objectManagerFrontendAttributes.api.ts'
- import { TicketCreateDocument } from '#shared/entities/ticket/graphql/mutations/create.api.ts'
- import { defaultOrganization } from '#mobile/entities/organization/__tests__/mocks/organization-mocks.ts'
- import {
- ticketObjectAttributes,
- ticketArticleObjectAttributes,
- ticketPayload,
- } from '#mobile/entities/ticket/__tests__/mocks/ticket-mocks.ts'
- const visitTicketCreate = async (path = '/tickets/create') => {
- const mockObjectAttributes = mockGraphQLApi(
- ObjectManagerFrontendAttributesDocument,
- ).willBehave(({ object }) => {
- if (object === 'Ticket') {
- return {
- data: {
- objectManagerFrontendAttributes: ticketObjectAttributes(),
- },
- }
- }
- return {
- data: {
- objectManagerFrontendAttributes: ticketArticleObjectAttributes(),
- },
- }
- })
- const mockFormUpdater = mockGraphQLApi(FormUpdaterDocument).willResolve({
- formUpdater: {
- fields: {
- ticket_duplicate_detection: {
- show: true,
- hidden: false,
- value: { count: 0, items: [] },
- },
- group_id: {
- show: true,
- options: [
- {
- label: 'Users',
- value: 1,
- },
- ],
- clearable: true,
- },
- owner_id: {
- show: true,
- options: [{ value: 100, label: 'Max Mustermann' }],
- },
- priority_id: {
- show: true,
- options: [
- { value: 1, label: '1 low' },
- { value: 2, label: '2 normal' },
- { value: 3, label: '3 high' },
- ],
- clearable: true,
- },
- pending_time: {
- show: false,
- required: false,
- hidden: false,
- disabled: false,
- },
- state_id: {
- show: true,
- options: [
- { value: 4, label: 'closed' },
- { value: 2, label: 'open' },
- { value: 7, label: 'pending close' },
- { value: 3, label: 'pending reminder' },
- ],
- clearable: true,
- },
- },
- },
- })
- const view = await visitView(path)
- await flushPromises()
- await getNode('ticket-create')?.settled
- return { mockFormUpdater, mockObjectAttributes, view }
- }
- const mockTicketCreate = () => {
- return mockGraphQLApi(TicketCreateDocument).willResolve({
- ticketCreate: {
- ticket: ticketPayload(),
- errors: null,
- __typename: 'TicketCreatePayload',
- },
- })
- }
- const mockCustomerQueryResult = () => {
- return mockGraphQLApi(AutocompleteSearchUserDocument).willResolve({
- autocompleteSearchUser: [
- nullableMock({
- value: '2',
- label: 'Nicole Braun',
- labelPlaceholder: null,
- heading: 'Zammad Foundation',
- headingPlaceholder: null,
- disabled: null,
- icon: null,
- user: {
- id: 'gid://zammad/User/2',
- internalId: 2,
- firstname: 'Nicole',
- lastname: 'Braun',
- fullname: 'Nicole Braun',
- image: null,
- objectAttributeValues: [],
- organization: {
- id: 'gid://zammad/Organization/1',
- internalId: 1,
- name: 'Zammad Foundation',
- active: true,
- objectAttributeValues: [],
- __typename: 'Organization',
- },
- hasSecondaryOrganizations: false,
- __typename: 'User',
- },
- __typename: 'AutocompleteSearchUserEntry',
- }),
- ],
- })
- }
- const nextStep = async (view: ExtendedRenderResult) => {
- await view.events.click(view.getByRole('button', { name: 'Continue' }))
- }
- const checkShownSteps = async (
- view: ExtendedRenderResult,
- steps: Array<string>,
- ) => {
- steps.forEach((step) => {
- expect(view.getByRole('button', { name: step })).toBeInTheDocument()
- })
- }
- beforeAll(async () => {
- // So we don't need to wait until it loads inside test.
- await import(
- '#shared/components/Form/fields/FieldEditor/FieldEditorInput.vue'
- )
- })
- describe('Creating new ticket as agent', () => {
- beforeEach(() => {
- mockPermissions(['ticket.agent'])
- mockApplicationConfig({
- customer_ticket_create: true,
- ui_ticket_create_available_types: ['phone-in', 'phone-out', 'email-out'],
- ui_ticket_create_default_type: 'phone-in',
- })
- })
- it('shows 4 steps for agents', async () => {
- const { view } = await visitTicketCreate()
- checkShownSteps(view, ['1', '2', '3', '4'])
- })
- it('disables the submit button if required data is missing', async () => {
- const { view } = await visitTicketCreate()
- expect(view.getByRole('button', { name: 'Create' })).toBeDisabled()
- })
- it('invalidates a single step if required data is missing', async () => {
- const { mockFormUpdater, view } = await visitTicketCreate()
- await view.events.type(view.getByLabelText('Title'), 'Foobar')
- await waitUntil(() => mockFormUpdater.calls.resolve)
- await nextStep(view)
- await nextStep(view)
- await nextStep(view)
- expect(
- view.getByRole('status', { name: 'Invalid values in step 3' }),
- ).toBeInTheDocument()
- })
- it.each([
- { name: 'Create', button: 'header submit button' },
- { name: 'Create ticket', button: 'footer submit button' },
- ])(
- 'redirects to detail view after successful ticket creation when clicked on $button',
- async ({ name }) => {
- const mockCustomer = mockCustomerQueryResult()
- const mockTicket = mockTicketCreate()
- const { mockFormUpdater, view } = await visitTicketCreate()
- await view.events.type(view.getByLabelText('Title'), 'Ticket Title')
- await waitUntil(() => mockFormUpdater.calls.resolve === 2)
- await nextStep(view)
- await nextStep(view)
- // Customer selection.
- await view.events.click(view.getByLabelText('Customer'))
- await view.events.type(await view.findByRole('searchbox'), 'nicole')
- await waitUntil(() => mockCustomer.calls.resolve)
- await view.events.click(view.getByText('Nicole Braun'))
- await waitUntil(() => mockFormUpdater.calls.resolve === 3)
- // Group selection.
- await view.events.click(view.getByLabelText('Group'))
- await view.events.click(view.getByText('Users'))
- await waitUntil(() => mockFormUpdater.calls.resolve === 4)
- await nextStep(view)
- // Text input.
- const editorNode = getNode('ticket-create')?.find('body', 'name')
- await editorNode?.input('Article body', false)
- // There is a button with "Create" in the header, and a "Create ticket" button in the footer.
- const submitButton = view.getByRole('button', { name })
- await waitUntil(() => !submitButton.hasAttribute('disabled'))
- expect(submitButton).not.toBeDisabled()
- // don't actually redirect
- const router = getTestRouter()
- router.mockMethods()
- await view.events.click(submitButton)
- await waitUntil(() => mockTicket.calls.resolve)
- expect(await view.findByRole('alert')).toHaveTextContent(
- 'Ticket has been created successfully.',
- )
- expect(router.replace).toHaveBeenCalledWith('/tickets/1')
- },
- )
- it('shows confirm popup, when leaving', async () => {
- const { view } = await visitTicketCreate()
- await view.events.type(view.getByLabelText('Title'), 'Foobar')
- // Wait on the changes
- await getNode('ticket-create')?.settled
- await view.events.click(view.getByRole('button', { name: 'Go home' }))
- expect(view.queryByTestId('popupWindow')).toBeInTheDocument()
- await expect(view.findByText('Confirm dialog')).resolves.toBeInTheDocument()
- })
- it('shows the CC field for type "Email"', async () => {
- const { mockFormUpdater, view } = await visitTicketCreate()
- await view.events.type(view.getByLabelText('Title'), 'Foobar')
- await waitUntil(() => mockFormUpdater.calls.resolve)
- await nextStep(view)
- await view.events.click(view.getByLabelText('Send Email'))
- await nextStep(view)
- expect(view.getByLabelText('CC')).toBeInTheDocument()
- })
- // The rest of the test cases are covered by E2E test, due to limitations of JSDOM test environment.
- })
- describe('Creating new ticket as customer', () => {
- beforeEach(() => {
- mockPermissions(['ticket.customer'])
- mockApplicationConfig({
- customer_ticket_create: true,
- })
- })
- it('shows 3 steps for customers', async () => {
- const { view } = await visitTicketCreate()
- checkShownSteps(view, ['1', '2', '3'])
- expect(view.queryByRole('button', { name: '4' })).not.toBeInTheDocument()
- })
- it('redirects to the error page if ticket creation is turned off', async () => {
- mockApplicationConfig({
- customer_ticket_create: false,
- })
- const { view } = await visitTicketCreate()
- expect(view.getByRole('main')).toHaveTextContent(
- 'Creating new tickets via web is disabled.',
- )
- })
- it('does not show the organization field without secondary organizations', async () => {
- mockUserCurrent({
- lastname: 'Doe',
- firstname: 'John',
- organization: defaultOrganization(),
- hasSecondaryOrganizations: false,
- })
- mockPermissions(['ticket.customer'])
- const { mockFormUpdater, view } = await visitTicketCreate()
- await view.events.type(view.getByLabelText('Title'), 'Foobar')
- await waitUntil(() => mockFormUpdater.calls.resolve)
- await nextStep(view)
- expect(view.queryByLabelText('Organization')).not.toBeInTheDocument()
- })
- it('does show the organization field with secondary organizations', async () => {
- mockUserCurrent({
- lastname: 'Doe',
- firstname: 'John',
- organization: defaultOrganization(),
- hasSecondaryOrganizations: true,
- })
- mockPermissions(['ticket.customer'])
- const { mockFormUpdater, view } = await visitTicketCreate()
- await view.events.type(view.getByLabelText('Title'), 'Foobar')
- await waitUntil(() => mockFormUpdater.calls.resolve)
- await nextStep(view)
- expect(view.queryByLabelText('Organization')).toBeInTheDocument()
- })
- it("doesn't show 'are you sure' dialog if duplicate protection is enabled and no changes were done", async () => {
- mockTicketOverviews()
- const { view } = await visitTicketCreate()
- await view.events.click(view.getByLabelText('Go home'))
- await waitFor(() => {
- expect(view.queryByText('Additional information')).not.toBeInTheDocument()
- })
- expect(view.queryByText('Confirm dialog')).not.toBeInTheDocument()
- })
- })
- describe('Creating new ticket as user having customer & agent permissions', () => {
- beforeEach(() => {
- mockPermissions(['ticket.agent', 'ticket.customer'])
- mockApplicationConfig({
- customer_ticket_create: true,
- ui_ticket_create_available_types: ['phone-in', 'phone-out', 'email-out'],
- ui_ticket_create_default_type: 'phone-in',
- })
- })
- it('does show the form for agents if having ticket.customer & ticket.agent permissions', async () => {
- const { view } = await visitTicketCreate()
- checkShownSteps(view, ['1', '2', '3', '4'])
- })
- })
- describe('Create ticket page redirects back', () => {
- it('correctly redirects from ticket create hash-based routes', async () => {
- setupView('agent')
- await visitTicketCreate('/#ticket/create')
- const router = getTestRouter()
- const route = router.currentRoute.value
- expect(route.name).toBe('TicketCreate')
- })
- it('correctly redirects from ticket create with id hash-based routes', async () => {
- setupView('agent')
- await visitTicketCreate('/#ticket/create/id/13214124')
- const router = getTestRouter()
- const route = router.currentRoute.value
- expect(route.name).toBe('TicketCreate')
- })
- })
|