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('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_login_at: '2026-04-16T01:00:00Z', 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: `
{{ columns.map(col => col.key).join(',') }}
` } 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 and used activity columns, hides last_login_at, and requests last_used_at sort', async () => { const wrapper = mount(UsersView, { global: { stubs: { AppLayout: { template: '
' }, TablePageLayout: { template: '
' }, 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() expect(columns).toContain('last_used_at') expect(columns).toContain('last_active_at') expect(columns).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) ) }) })