123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223 |
- // Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- import { getAllByRole, getByRole, queryByRole } from '@testing-library/vue'
- import { flushPromises } from '@vue/test-utils'
- import { visitView } from '#tests/support/components/visitView.ts'
- import { mockApplicationConfig } from '#tests/support/mock-applicationConfig.ts'
- import { mockAuthentication } from '#tests/support/mock-authentication.ts'
- import { mockPermissions } from '#tests/support/mock-permissions.ts'
- import { mockFormUpdaterQuery } from '#shared/components/Form/graphql/queries/formUpdater.mocks.ts'
- import {
- mockUserAddMutation,
- waitForUserAddMutationCalls,
- } from '#shared/entities/user/graphql/mutations/add.mocks.ts'
- import { EnumSystemSetupInfoStatus } from '#shared/graphql/types.ts'
- import { mockSystemSetupInfoQuery } from '../graphql/queries/systemSetupInfo.mocks.ts'
- describe('guided setup manual invite', () => {
- describe('when system is not ready', () => {
- beforeEach(() => {
- mockApplicationConfig({
- system_init_done: false,
- })
- })
- it('redirects to guided setup start', async () => {
- mockSystemSetupInfoQuery({
- systemSetupInfo: {
- status: EnumSystemSetupInfoStatus.New,
- type: null,
- },
- })
- const view = await visitView('/guided-setup/manual/invite')
- await vi.waitFor(() => {
- expect(
- view,
- 'correctly redirects to guided setup start screen',
- ).toHaveCurrentUrl('/guided-setup')
- })
- view.getByText('Set up a new system')
- })
- })
- describe('when system is ready for optional steps', () => {
- beforeEach(() => {
- mockApplicationConfig({
- system_init_done: true,
- })
- mockPermissions(['admin'])
- mockAuthentication(true)
- mockFormUpdaterQuery({
- formUpdater: {
- fields: {
- role_ids: {
- initialValue: [2],
- options: [
- {
- value: 1,
- label: 'Admin',
- description: 'To configure your system.',
- },
- {
- value: 2,
- label: 'Agent',
- description: 'To work on Tickets.',
- },
- {
- value: 3,
- label: 'Customer',
- description: 'People who create Tickets ask for help.',
- },
- ],
- },
- group_ids: {
- options: [
- {
- value: 1,
- label: 'Users',
- },
- {
- value: 2,
- label: 'some group1',
- },
- ],
- },
- },
- },
- })
- })
- it('can invite user and rerender the form', async () => {
- const view = await visitView('/guided-setup/manual/invite')
- await flushPromises()
- expect(view.getByText('Invite Colleagues')).toBeInTheDocument()
- expect(view.getByLabelText('First name')).toBeInTheDocument()
- expect(view.getByLabelText('Last name')).toBeInTheDocument()
- expect(view.getByLabelText('Email')).toBeInTheDocument()
- expect(view.getByLabelText('Roles')).toBeInTheDocument()
- expect(view.getByLabelText('Group permissions')).toBeInTheDocument()
- expect(
- view.getByRole('button', { name: 'Finish Setup' }),
- ).toBeInTheDocument()
- await view.events.type(view.getByLabelText('Email'), 'test@example.com')
- await view.events.click(view.getAllByRole('switch')[2])
- const groupPermissions = view.getByLabelText('Group permissions')
- const combobox = getByRole(groupPermissions, 'combobox')
- await view.events.click(combobox)
- const listbox = view.getByRole('listbox')
- const options = getAllByRole(listbox, 'option')
- await view.events.click(options[0])
- await view.events.click(view.getByLabelText('Full'))
- expect(view.getByLabelText('Email')).toHaveValue('test@example.com')
- expect(view.getAllByRole('switch')[2]).toHaveAttribute(
- 'aria-checked',
- 'true',
- )
- expect(queryByRole(combobox, 'listitem')).toBeInTheDocument()
- expect(view.getByLabelText('Full')).toBeChecked()
- const inviteButton = view.getByRole('button', {
- name: 'Send Invitation',
- })
- await view.events.click(inviteButton)
- const calls = await waitForUserAddMutationCalls()
- expect(calls.at(-1)?.variables).toEqual({
- input: {
- firstname: '',
- lastname: '',
- email: 'test@example.com',
- roleIds: ['gid://zammad/Role/2', 'gid://zammad/Role/3'],
- groupIds: [
- {
- groupInternalId: 1,
- accessType: ['full'],
- },
- ],
- objectAttributeValues: [],
- },
- sendInvite: true,
- })
- expect(await view.findByText('Invitation sent!')).toBeInTheDocument()
- expect(view.getByLabelText('Email')).toHaveValue('')
- expect(view.getAllByRole('switch')[2]).not.toHaveAttribute(
- 'aria-checked',
- 'true',
- )
- expect(queryByRole(combobox, 'listitem')).not.toBeInTheDocument()
- expect(view.getByLabelText('Full')).not.toBeChecked()
- expect(
- view,
- 'stays on the guided setup manual invite step',
- ).toHaveCurrentUrl('/guided-setup/manual/invite')
- })
- it('can display form errors', async () => {
- const view = await visitView('/guided-setup/manual/invite')
- mockUserAddMutation({
- userAdd: {
- user: null,
- errors: [
- {
- message:
- 'At least one identifier (firstname, lastname, phone or email) for user is required.',
- field: null,
- },
- ],
- },
- })
- const inviteButton = view.getByRole('button', {
- name: 'Send Invitation',
- })
- await view.events.click(inviteButton)
- expect(
- view.getByText(
- 'At least one identifier (firstname, lastname, phone or email) for user is required.',
- ),
- ).toHaveRole('alert')
- })
- it('redirects to guided setup finish screen when continued', async () => {
- const view = await visitView('/guided-setup/manual/invite')
- await view.events.click(
- view.getByRole('button', { name: 'Finish Setup' }),
- )
- await vi.waitFor(() => {
- expect(
- view,
- 'correctly redirects to guided setup finish screen',
- ).toHaveCurrentUrl('/guided-setup/manual/finish')
- })
- })
- })
- })
|