165 lines
4.2 KiB
TypeScript
165 lines
4.2 KiB
TypeScript
import { beforeEach, describe, expect, it, vi } from 'vitest'
|
|
import { flushPromises, mount } from '@vue/test-utils'
|
|
|
|
import type { AdminUser } from '@/types'
|
|
import UsersView from '../UsersView.vue'
|
|
|
|
const {
|
|
listUsers,
|
|
getAllGroups,
|
|
getBatchUsersUsage,
|
|
listEnabledDefinitions,
|
|
getBatchUserAttributes
|
|
} = vi.hoisted(() => ({
|
|
listUsers: vi.fn(),
|
|
getAllGroups: vi.fn(),
|
|
getBatchUsersUsage: vi.fn(),
|
|
listEnabledDefinitions: vi.fn(),
|
|
getBatchUserAttributes: vi.fn()
|
|
}))
|
|
|
|
vi.mock('@/api/admin', () => ({
|
|
adminAPI: {
|
|
users: {
|
|
list: listUsers,
|
|
toggleStatus: vi.fn(),
|
|
delete: vi.fn()
|
|
},
|
|
groups: {
|
|
getAll: getAllGroups
|
|
},
|
|
dashboard: {
|
|
getBatchUsersUsage
|
|
},
|
|
userAttributes: {
|
|
listEnabledDefinitions,
|
|
getBatchUserAttributes
|
|
}
|
|
}
|
|
}))
|
|
|
|
vi.mock('@/stores/app', () => ({
|
|
useAppStore: () => ({
|
|
showError: vi.fn(),
|
|
showSuccess: vi.fn()
|
|
})
|
|
}))
|
|
|
|
vi.mock('vue-i18n', async () => {
|
|
const actual = await vi.importActual<typeof import('vue-i18n')>('vue-i18n')
|
|
return {
|
|
...actual,
|
|
useI18n: () => ({
|
|
t: (key: string) => key
|
|
})
|
|
}
|
|
})
|
|
|
|
const createAdminUser = (): AdminUser => ({
|
|
id: 42,
|
|
username: 'scoped-user',
|
|
email: 'scoped@example.com',
|
|
role: 'user',
|
|
balance: 0,
|
|
concurrency: 1,
|
|
status: 'active',
|
|
allowed_groups: [],
|
|
balance_notify_enabled: false,
|
|
balance_notify_threshold: null,
|
|
balance_notify_extra_emails: [],
|
|
created_at: '2026-04-17T00:00:00Z',
|
|
updated_at: '2026-04-17T00:00:00Z',
|
|
notes: '',
|
|
last_active_at: '2026-04-16T02:00:00Z',
|
|
last_used_at: '2026-04-17T02:00:00Z',
|
|
current_concurrency: 0
|
|
})
|
|
|
|
const DataTableStub = {
|
|
props: ['columns', 'data'],
|
|
emits: ['sort'],
|
|
template: `
|
|
<div>
|
|
<div data-test="columns">{{ columns.map(col => col.key).join(',') }}</div>
|
|
<button data-test="sort-last-used" @click="$emit('sort', 'last_used_at', 'desc')">sort</button>
|
|
<div v-for="row in data" :key="row.id">
|
|
<slot name="cell-last_used_at" :value="row.last_used_at" :row="row" />
|
|
</div>
|
|
</div>
|
|
`
|
|
}
|
|
|
|
describe('admin UsersView', () => {
|
|
beforeEach(() => {
|
|
localStorage.clear()
|
|
|
|
listUsers.mockReset()
|
|
getAllGroups.mockReset()
|
|
getBatchUsersUsage.mockReset()
|
|
listEnabledDefinitions.mockReset()
|
|
getBatchUserAttributes.mockReset()
|
|
|
|
listUsers.mockResolvedValue({
|
|
items: [createAdminUser()],
|
|
total: 1,
|
|
page: 1,
|
|
page_size: 20,
|
|
pages: 1
|
|
})
|
|
getAllGroups.mockResolvedValue([])
|
|
getBatchUsersUsage.mockResolvedValue({ stats: {} })
|
|
listEnabledDefinitions.mockResolvedValue([])
|
|
getBatchUserAttributes.mockResolvedValue({ values: {} })
|
|
})
|
|
|
|
it('shows active, used, and created activity columns in order and requests last_used_at sort', async () => {
|
|
const wrapper = mount(UsersView, {
|
|
global: {
|
|
stubs: {
|
|
AppLayout: { template: '<div><slot /></div>' },
|
|
TablePageLayout: {
|
|
template: '<div><slot name="filters" /><slot name="table" /><slot name="pagination" /></div>'
|
|
},
|
|
DataTable: DataTableStub,
|
|
Pagination: true,
|
|
ConfirmDialog: true,
|
|
EmptyState: true,
|
|
GroupBadge: true,
|
|
Select: true,
|
|
UserAttributesConfigModal: true,
|
|
UserConcurrencyCell: true,
|
|
UserCreateModal: true,
|
|
UserEditModal: true,
|
|
UserApiKeysModal: true,
|
|
UserAllowedGroupsModal: true,
|
|
UserBalanceModal: true,
|
|
UserBalanceHistoryModal: true,
|
|
GroupReplaceModal: true,
|
|
Icon: true,
|
|
Teleport: true
|
|
}
|
|
}
|
|
})
|
|
|
|
await flushPromises()
|
|
|
|
const columns = wrapper.get('[data-test="columns"]').text()
|
|
const visibleColumns = columns.split(',')
|
|
expect(visibleColumns.slice(-4, -1)).toEqual(['last_active_at', 'last_used_at', 'created_at'])
|
|
expect(visibleColumns).not.toContain('last_login_at')
|
|
|
|
await wrapper.get('[data-test="sort-last-used"]').trigger('click')
|
|
await flushPromises()
|
|
|
|
expect(listUsers).toHaveBeenLastCalledWith(
|
|
1,
|
|
20,
|
|
expect.objectContaining({
|
|
sort_by: 'last_used_at',
|
|
sort_order: 'desc'
|
|
}),
|
|
expect.any(Object)
|
|
)
|
|
})
|
|
})
|