mirror of
https://github.com/Comfy-Org/ComfyUI_frontend.git
synced 2026-02-04 07:00:23 +00:00
239 lines
6.5 KiB
TypeScript
239 lines
6.5 KiB
TypeScript
import type { VueWrapper } from '@vue/test-utils'
|
|
import { mount } from '@vue/test-utils'
|
|
import Button from 'primevue/button'
|
|
import { afterAll, beforeEach, describe, expect, it, vi } from 'vitest'
|
|
import { h } from 'vue'
|
|
import { createI18n } from 'vue-i18n'
|
|
|
|
import enMessages from '@/locales/en/main.json' with { type: 'json' }
|
|
|
|
import CurrentUserPopover from './CurrentUserPopover.vue'
|
|
|
|
// Mock all firebase modules
|
|
vi.mock('firebase/app', () => ({
|
|
initializeApp: vi.fn(),
|
|
getApp: vi.fn()
|
|
}))
|
|
|
|
vi.mock('firebase/auth', () => ({
|
|
getAuth: vi.fn(),
|
|
setPersistence: vi.fn(),
|
|
browserLocalPersistence: {},
|
|
onAuthStateChanged: vi.fn(),
|
|
signInWithEmailAndPassword: vi.fn(),
|
|
signOut: vi.fn()
|
|
}))
|
|
|
|
// Mock pinia
|
|
vi.mock('pinia')
|
|
|
|
// Mock showSettingsDialog and showTopUpCreditsDialog
|
|
const mockShowSettingsDialog = vi.fn()
|
|
const mockShowTopUpCreditsDialog = vi.fn()
|
|
|
|
// Mock window.open
|
|
const originalWindowOpen = window.open
|
|
beforeEach(() => {
|
|
window.open = vi.fn()
|
|
})
|
|
|
|
afterAll(() => {
|
|
window.open = originalWindowOpen
|
|
})
|
|
|
|
// Mock the useCurrentUser composable
|
|
const mockHandleSignOut = vi.fn()
|
|
vi.mock('@/composables/auth/useCurrentUser', () => ({
|
|
useCurrentUser: vi.fn(() => ({
|
|
userPhotoUrl: 'https://example.com/avatar.jpg',
|
|
userDisplayName: 'Test User',
|
|
userEmail: 'test@example.com',
|
|
handleSignOut: mockHandleSignOut
|
|
}))
|
|
}))
|
|
|
|
// Mock the useFirebaseAuthActions composable
|
|
const mockLogout = vi.fn()
|
|
vi.mock('@/composables/auth/useFirebaseAuthActions', () => ({
|
|
useFirebaseAuthActions: vi.fn(() => ({
|
|
fetchBalance: vi.fn().mockResolvedValue(undefined),
|
|
logout: mockLogout
|
|
}))
|
|
}))
|
|
|
|
// Mock the dialog service
|
|
vi.mock('@/services/dialogService', () => ({
|
|
useDialogService: vi.fn(() => ({
|
|
showSettingsDialog: mockShowSettingsDialog,
|
|
showTopUpCreditsDialog: mockShowTopUpCreditsDialog
|
|
}))
|
|
}))
|
|
|
|
// Mock the firebaseAuthStore
|
|
vi.mock('@/stores/firebaseAuthStore', () => ({
|
|
useFirebaseAuthStore: vi.fn(() => ({
|
|
getAuthHeader: vi
|
|
.fn()
|
|
.mockResolvedValue({ Authorization: 'Bearer mock-token' })
|
|
}))
|
|
}))
|
|
|
|
// Mock the useSubscription composable
|
|
const mockFetchStatus = vi.fn().mockResolvedValue(undefined)
|
|
vi.mock('@/platform/cloud/subscription/composables/useSubscription', () => ({
|
|
useSubscription: vi.fn(() => ({
|
|
isSubscriptionRequirementMet: { value: true },
|
|
fetchStatus: mockFetchStatus
|
|
}))
|
|
}))
|
|
|
|
// Mock UserAvatar component
|
|
vi.mock('@/components/common/UserAvatar.vue', () => ({
|
|
default: {
|
|
name: 'UserAvatarMock',
|
|
render() {
|
|
return h('div', 'Avatar')
|
|
}
|
|
}
|
|
}))
|
|
|
|
// Mock UserCredit component
|
|
vi.mock('@/components/common/UserCredit.vue', () => ({
|
|
default: {
|
|
name: 'UserCreditMock',
|
|
render() {
|
|
return h('div', 'Credit: 100')
|
|
}
|
|
}
|
|
}))
|
|
|
|
vi.mock('@/platform/cloud/subscription/components/SubscribeButton.vue', () => ({
|
|
default: {
|
|
name: 'SubscribeButtonMock',
|
|
render() {
|
|
return h('div', 'Subscribe Button')
|
|
}
|
|
}
|
|
}))
|
|
|
|
describe('CurrentUserPopover', () => {
|
|
beforeEach(() => {
|
|
vi.clearAllMocks()
|
|
})
|
|
|
|
const findButtonByLabel = (wrapper: VueWrapper, label: string) =>
|
|
wrapper
|
|
.findAllComponents(Button)
|
|
.find((button) => button.props('label') === label)!
|
|
|
|
const mountComponent = (): VueWrapper => {
|
|
const i18n = createI18n({
|
|
legacy: false,
|
|
locale: 'en',
|
|
messages: { en: enMessages }
|
|
})
|
|
|
|
return mount(CurrentUserPopover, {
|
|
global: {
|
|
plugins: [i18n],
|
|
stubs: {
|
|
Divider: true
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
it('renders user information correctly', () => {
|
|
const wrapper = mountComponent()
|
|
|
|
expect(wrapper.text()).toContain('Test User')
|
|
expect(wrapper.text()).toContain('test@example.com')
|
|
})
|
|
|
|
it('renders logout button with correct props', () => {
|
|
const wrapper = mountComponent()
|
|
|
|
// Find all buttons and get the logout button (last button)
|
|
const buttons = wrapper.findAllComponents(Button)
|
|
const logoutButton = buttons[4]
|
|
|
|
// Check that logout button has correct props
|
|
expect(logoutButton.props('label')).toBe('Log Out')
|
|
expect(logoutButton.props('icon')).toBe('pi pi-sign-out')
|
|
})
|
|
|
|
it('opens user settings and emits close event when settings button is clicked', async () => {
|
|
const wrapper = mountComponent()
|
|
|
|
// Find all buttons and get the settings button (third button)
|
|
const settingsButton = findButtonByLabel(wrapper, 'User Settings')
|
|
|
|
// Click the settings button
|
|
await settingsButton.trigger('click')
|
|
|
|
// Verify showSettingsDialog was called with 'user'
|
|
expect(mockShowSettingsDialog).toHaveBeenCalledWith('user')
|
|
|
|
// Verify close event was emitted
|
|
expect(wrapper.emitted('close')).toBeTruthy()
|
|
expect(wrapper.emitted('close')!.length).toBe(1)
|
|
})
|
|
|
|
it('calls logout function and emits close event when logout button is clicked', async () => {
|
|
const wrapper = mountComponent()
|
|
|
|
// Find all buttons and get the logout button (last button)
|
|
const logoutButton = findButtonByLabel(wrapper, 'Log Out')
|
|
|
|
// Click the logout button
|
|
await logoutButton.trigger('click')
|
|
|
|
// Verify handleSignOut was called
|
|
expect(mockHandleSignOut).toHaveBeenCalled()
|
|
|
|
// Verify close event was emitted
|
|
expect(wrapper.emitted('close')).toBeTruthy()
|
|
expect(wrapper.emitted('close')!.length).toBe(1)
|
|
})
|
|
|
|
it('opens API pricing docs and emits close event when API pricing button is clicked', async () => {
|
|
const wrapper = mountComponent()
|
|
|
|
// Find all buttons and get the Partner Nodes info button (first one)
|
|
const partnerNodesButton = findButtonByLabel(
|
|
wrapper,
|
|
'Partner Nodes pricing table'
|
|
)
|
|
|
|
// Click the Partner Nodes button
|
|
await partnerNodesButton.trigger('click')
|
|
|
|
// Verify window.open was called with the correct URL
|
|
expect(window.open).toHaveBeenCalledWith(
|
|
'https://docs.comfy.org/tutorials/api-nodes/overview#api-nodes',
|
|
'_blank'
|
|
)
|
|
|
|
// Verify close event was emitted
|
|
expect(wrapper.emitted('close')).toBeTruthy()
|
|
expect(wrapper.emitted('close')!.length).toBe(1)
|
|
})
|
|
|
|
it('opens top-up dialog and emits close event when top-up button is clicked', async () => {
|
|
const wrapper = mountComponent()
|
|
|
|
// Find all buttons and get the top-up button (second one)
|
|
const topUpButton = findButtonByLabel(wrapper, 'Top Up')
|
|
|
|
// Click the top-up button
|
|
await topUpButton.trigger('click')
|
|
|
|
// Verify showTopUpCreditsDialog was called
|
|
expect(mockShowTopUpCreditsDialog).toHaveBeenCalled()
|
|
|
|
// Verify close event was emitted
|
|
expect(wrapper.emitted('close')).toBeTruthy()
|
|
expect(wrapper.emitted('close')!.length).toBe(1)
|
|
})
|
|
})
|