Files
sub2api/frontend/src/components/user/profile/__tests__/ProfileInfoCard.spec.ts

198 lines
5.9 KiB
TypeScript

import { mount } from '@vue/test-utils'
import { describe, expect, it, vi } from 'vitest'
import ProfileInfoCard from '@/components/user/profile/ProfileInfoCard.vue'
import type { User } from '@/types'
vi.mock('vue-router', () => ({
useRoute: () => ({
fullPath: '/profile'
})
}))
vi.mock('@/stores/auth', () => ({
useAuthStore: () => ({
user: null
})
}))
vi.mock('@/stores/app', () => ({
useAppStore: () => ({
showError: vi.fn(),
showSuccess: vi.fn()
})
}))
vi.mock('vue-i18n', async (importOriginal) => {
const actual = await importOriginal<typeof import('vue-i18n')>()
return {
...actual,
useI18n: () => ({
t: (key: string, params?: Record<string, string>) => {
if (key === 'profile.accountBalance') return 'Account Balance'
if (key === 'profile.concurrencyLimit') return 'Concurrency Limit'
if (key === 'profile.memberSince') return 'Member Since'
if (key === 'profile.administrator') return 'Administrator'
if (key === 'profile.user') return 'User'
if (key === 'profile.authBindings.providers.email') return 'Email'
if (key === 'profile.authBindings.providers.linuxdo') return 'LinuxDo'
if (key === 'profile.authBindings.providers.wechat') return 'WeChat'
if (key === 'profile.authBindings.providers.oidc') return params?.providerName || 'OIDC'
if (key === 'profile.authBindings.source.avatar') {
return `Avatar synced from ${params?.providerName || 'provider'}`
}
if (key === 'profile.authBindings.source.username') {
return `Username synced from ${params?.providerName || 'provider'}`
}
return key
}
})
}
})
function createUser(overrides: Partial<User> = {}): User {
return {
id: 5,
username: 'alice',
email: 'alice@example.com',
avatar_url: null,
role: 'user',
balance: 10,
concurrency: 2,
status: 'active',
allowed_groups: null,
balance_notify_enabled: true,
balance_notify_threshold: null,
balance_notify_extra_emails: [],
created_at: '2026-04-20T00:00:00Z',
updated_at: '2026-04-20T00:00:00Z',
...overrides
}
}
describe('ProfileInfoCard', () => {
it('renders basic account information inside the new overview shell', () => {
const wrapper = mount(ProfileInfoCard, {
props: {
user: createUser()
},
global: {
stubs: {
Icon: true
}
}
})
expect(wrapper.text()).toContain('alice@example.com')
expect(wrapper.text()).toContain('alice')
expect(wrapper.text()).toContain('User')
expect(wrapper.get('[data-testid="profile-basics-panel"]').exists()).toBe(true)
expect(wrapper.get('[data-testid="profile-auth-bindings-panel"]').exists()).toBe(true)
})
it('renders third-party source hints from profile sources', () => {
const wrapper = mount(ProfileInfoCard, {
props: {
user: createUser({
avatar_url: 'https://cdn.example.com/linuxdo.png',
profile_sources: {
avatar: { provider: 'linuxdo', source: 'linuxdo' },
username: { provider: 'linuxdo', source: 'linuxdo' }
}
})
},
global: {
stubs: {
Icon: true
}
}
})
expect(wrapper.text()).toContain('Avatar synced from LinuxDo')
expect(wrapper.text()).toContain('Username synced from LinuxDo')
})
it('uses the configured OIDC provider name in source hints', () => {
const wrapper = mount(ProfileInfoCard, {
props: {
user: createUser({
profile_sources: {
username: { provider: 'oidc', source: 'oidc' }
}
}),
oidcProviderName: 'ExampleID'
},
global: {
stubs: {
Icon: true
}
}
})
expect(wrapper.text()).toContain('Username synced from ExampleID')
})
it('does not display synthetic oauth-only emails as a real bound email', () => {
const wrapper = mount(ProfileInfoCard, {
props: {
user: createUser({
email: 'legacy-user@oidc-connect.invalid',
email_bound: false,
auth_bindings: {
email: { bound: false }
}
})
},
global: {
stubs: {
Icon: true
}
}
})
expect(wrapper.text()).not.toContain('legacy-user@oidc-connect.invalid')
})
it('does not display synthetic oauth-only emails when only legacy identity bindings mark email as unbound', () => {
const wrapper = mount(ProfileInfoCard, {
props: {
user: createUser({
email: 'legacy-user@wechat-connect.invalid',
identity_bindings: {
email: { bound: false }
}
})
},
global: {
stubs: {
Icon: true
}
}
})
expect(wrapper.text()).not.toContain('legacy-user@wechat-connect.invalid')
})
it('renders the approved overview hero and two-column content shell', () => {
const wrapper = mount(ProfileInfoCard, {
props: {
user: createUser()
},
global: {
stubs: {
Icon: true
}
}
})
expect(wrapper.get('[data-testid="profile-overview-hero"]').text()).toContain('alice@example.com')
expect(wrapper.get('[data-testid="profile-overview-metric-balance"]').text()).toContain('Account Balance')
expect(wrapper.get('[data-testid="profile-overview-metric-concurrency"]').text()).toContain('Concurrency Limit')
expect(wrapper.get('[data-testid="profile-overview-metric-member-since"]').text()).toContain('Member Since')
expect(wrapper.find('[data-testid="profile-info-summary-grid"]').exists()).toBe(false)
expect(wrapper.get('[data-testid="profile-main-column"]').exists()).toBe(true)
expect(wrapper.get('[data-testid="profile-side-column"]').exists()).toBe(true)
expect(wrapper.get('[data-testid="profile-basics-panel"]').exists()).toBe(true)
expect(wrapper.get('[data-testid="profile-auth-bindings-panel"]').exists()).toBe(true)
})
})