123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249 |
- // Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- import { getTestRouter } from '#tests/support/components/renderComponent.ts'
- import { visitView } from '#tests/support/components/visitView.ts'
- import {
- mockGraphQLApi,
- mockGraphQLSubscription,
- } from '#tests/support/mock-graphql-api.ts'
- import { setupView } from '#tests/support/mock-user.ts'
- import { waitUntil, waitUntilApisResolved } from '#tests/support/utils.ts'
- import { UserDocument } from '#shared/entities/user/graphql/queries/user.api.ts'
- import { UserUpdatesDocument } from '#shared/graphql/subscriptions/userUpdates.api.ts'
- import {
- defaultUser,
- mockUserDetailsApis,
- } from '#mobile/entities/user/__tests__/mocks/user-mocks.ts'
- describe('visiting user page', () => {
- test('view static content', async () => {
- const { mockUser, mockAttributes, user } = mockUserDetailsApis()
- const view = await visitView(`/users/${user.internalId}`)
- await waitUntilApisResolved(mockUser, mockAttributes)
- const organization = user.organization!
- expect(
- view.getByRole('img', { name: 'Avatar (John Doe)' }),
- ).toBeInTheDocument()
- const organizationLink = view.getByRole('link', {
- name: organization.name!,
- })
- expect(organizationLink).toBeInTheDocument()
- expect(organizationLink).toHaveAttribute(
- 'href',
- `/mobile/organizations/${organization.internalId}`,
- )
- expect(
- view.queryByRole('region', { name: 'First name' }),
- ).not.toBeInTheDocument()
- expect(
- view.queryByRole('region', { name: 'Last name' }),
- ).not.toBeInTheDocument()
- const ticketOpenLink = view.getByRole('link', { name: 'open 4' })
- const ticketClosedLink = view.getByRole('link', { name: 'closed 2' })
- expect(ticketOpenLink).toHaveAttribute(
- 'href',
- expect.stringContaining(`customer.id: ${user.internalId}`),
- )
- expect(ticketClosedLink).toHaveAttribute(
- 'href',
- expect.stringContaining(`customer.id: ${user.internalId}`),
- )
- expect(view.getByText('Secondary organizations')).toBeInTheDocument()
- expect(view.getByText('Dammaz')).toBeInTheDocument()
- })
- test('can toggle tickets count view', async () => {
- const { mockUser, mockAttributes, user } = mockUserDetailsApis()
- const view = await visitView(`/users/${user.internalId}`)
- await waitUntilApisResolved(mockUser, mockAttributes)
- const ticketOpenLink = view.getByRole('link', { name: 'open 4' })
- const ticketClosedLink = view.getByRole('link', { name: 'closed 2' })
- expect(ticketOpenLink).toBeInTheDocument()
- expect(ticketClosedLink).toBeInTheDocument()
- await view.events.click(
- view.getByRole('tab', { name: 'Organization tickets' }),
- )
- const organization = user.organization!
- const ticketOrganizationOpenLink = view.getByRole('link', {
- name: 'open 3',
- })
- const ticketOrganizationClosedLink = view.getByRole('link', {
- name: 'closed 1',
- })
- expect(ticketOrganizationOpenLink).toHaveAttribute(
- 'href',
- expect.stringContaining(`organization.id: ${organization.internalId}`),
- )
- expect(ticketOrganizationClosedLink).toHaveAttribute(
- 'href',
- expect.stringContaining(`organization.id: ${organization.internalId}`),
- )
- expect(
- view.getByRole('link', {
- name: 'Create new ticket for this organization',
- }),
- ).toBeInTheDocument()
- await view.events.click(view.getByRole('tab', { name: 'Their tickets' }))
- expect(view.getByRole('link', { name: 'open 4' })).toBeInTheDocument()
- expect(view.getByRole('link', { name: 'closed 2' })).toBeInTheDocument()
- expect(
- view.getByRole('link', { name: 'Create new ticket for this user' }),
- ).toBeInTheDocument()
- })
- test('view user without organization', async () => {
- const user = defaultUser()
- user.organization = null
- const { mockUser, mockAttributes } = mockUserDetailsApis(user)
- const view = await visitView(`/users/${user.internalId}`)
- await waitUntilApisResolved(mockUser, mockAttributes)
- expect(view.queryByTestId('organization-link')).not.toBeInTheDocument()
- expect(
- view.queryByRole('button', { name: 'Their tickets' }),
- ).not.toBeInTheDocument()
- expect(
- view.queryByRole('button', { name: 'Organization tickets' }),
- ).not.toBeInTheDocument()
- })
- test('view fully configured user', async () => {
- const userDefault = defaultUser()
- const [department, address] = userDefault.objectAttributeValues!
- const user = {
- ...userDefault,
- image: '',
- email: 'some-email@mail.com',
- web: 'https://some-web.com',
- vip: true,
- outOfOffice: false,
- outOfOfficeStartAt: null,
- outOfOfficeEndAt: null,
- phone: '80542243532',
- mobile: '2432332143',
- fax: 'fax.fax',
- note: 'This user is cool',
- objectAttributeValues: [
- {
- ...department,
- value: 'Department of Health and Safety',
- },
- {
- ...address,
- value: 'Berlin',
- },
- ],
- }
- const { mockUser, mockAttributes } = mockUserDetailsApis(user)
- const view = await visitView(`/users/${user.internalId}`)
- await waitUntilApisResolved(mockUser, mockAttributes)
- const getRegion = (name: string) => view.getByRole('region', { name })
- expect(view.getByIconName('crown'), 'vip has crown').toBeInTheDocument()
- expect(
- view.queryByRole('region', { name: 'First name' }),
- ).not.toBeInTheDocument()
- expect(
- view.queryByRole('region', { name: 'Last name' }),
- ).not.toBeInTheDocument()
- expect(getRegion('Email')).toHaveTextContent('some-email@mail.com')
- expect(getRegion('Web')).toHaveTextContent('https://some-web.com')
- expect(getRegion('Phone')).toHaveTextContent('80542243532')
- expect(getRegion('Mobile')).toHaveTextContent('2432332143')
- expect(getRegion('Fax')).toHaveTextContent('fax.fax')
- expect(getRegion('Note')).toHaveTextContent('This user is cool')
- expect(getRegion('Department')).toHaveTextContent(
- 'Department of Health and Safety',
- )
- expect(getRegion('Address')).toHaveTextContent('Berlin')
- })
- it('can edit user with required update policy', async () => {
- const { mockUser, mockAttributes, user } = mockUserDetailsApis()
- const view = await visitView(`/users/${user.internalId}`)
- await waitUntilApisResolved(mockUser, mockAttributes)
- expect(view.getByRole('button', { name: 'Edit' })).toBeInTheDocument()
- })
- it('cannot edit user without required update policy', async () => {
- const user = defaultUser()
- user.policy.update = false
- const { mockUser, mockAttributes } = mockUserDetailsApis(user)
- const view = await visitView(`/users/${user.internalId}`)
- await waitUntilApisResolved(mockUser, mockAttributes)
- expect(view.queryByRole('button', { name: 'Edit' })).not.toBeInTheDocument()
- })
- it('redirects to error page if user is not found', async () => {
- setupView('agent')
- const mockApi = mockGraphQLApi(UserDocument).willFailWithNotFoundError()
- mockGraphQLSubscription(UserUpdatesDocument)
- const view = await visitView('/users/123')
- await waitUntil(() => mockApi.calls.error)
- await expect(view.findByText('Not Found')).resolves.toBeInTheDocument()
- })
- it('redirects to error page if access to organization is forbidden', async () => {
- setupView('agent')
- const mockApi = mockGraphQLApi(UserDocument).willFailWithForbiddenError()
- mockGraphQLSubscription(UserUpdatesDocument)
- const view = await visitView('/users/123')
- await waitUntil(() => mockApi.calls.error)
- await expect(view.findByText('Forbidden')).resolves.toBeInTheDocument()
- })
- })
- test('correctly redirects from hash-based routes', async () => {
- setupView('agent')
- mockGraphQLSubscription(UserUpdatesDocument)
- await visitView('/#user/profile/1')
- const router = getTestRouter()
- const route = router.currentRoute.value
- expect(route.name).toBe('UserDetailView')
- expect(route.params).toEqual({ internalId: '1' })
- })
|