Files
termi-blog/playwright-smoke/tests/admin.spec.ts
limitcool 27d0827f3e
All checks were successful
docker-images / resolve-build-targets (push) Successful in 4s
ui-regression / playwright-regression (push) Successful in 5m55s
docker-images / build-and-push (admin) (push) Successful in 54s
docker-images / build-and-push (backend) (push) Successful in 4s
docker-images / build-and-push (frontend) (push) Successful in 1m8s
docker-images / submit-indexnow (push) Successful in 15s
feat: 增强维护模式和审计页面功能,优化构建流程
2026-04-03 01:33:24 +08:00

386 lines
17 KiB
TypeScript

import { expect, test, type Page } from '@playwright/test'
import { getDebugState, loginAdmin, MOCK_BASE_URL, resetMockState } from './helpers'
test.beforeEach(async ({ request }) => {
await resetMockState(request)
})
function acceptNextDialog(page: Page) {
page.once('dialog', async (dialog) => {
await dialog.accept()
})
}
function buildSvgPayload(name: string, label: string) {
return {
name,
mimeType: 'image/svg+xml',
buffer: Buffer.from(
`<svg xmlns="http://www.w3.org/2000/svg" width="1200" height="675" viewBox="0 0 1200 675"><rect width="1200" height="675" fill="#111827"/><text x="80" y="180" fill="#f8fafc" font-family="monospace" font-size="40">${label}</text></svg>`,
'utf8',
),
}
}
test('后台登录、导航与关键模块页面可加载', async ({ page }) => {
await loginAdmin(page)
const routes = [
{ label: '概览', url: /\/$/, text: '运营总览' },
{ label: '数据分析', url: /\/analytics$/, text: '前台搜索、阅读行为与 AI 问答洞察' },
{ label: '文章', url: /\/posts$/, text: '文章列表' },
{ label: '分类', url: /\/categories$/, text: '分类目录' },
{ label: '标签', url: /\/tags$/, text: '标签库' },
{ label: '备份', url: /\/backups$/, text: '全站内容备份' },
{ label: '版本', url: /\/revisions$/, text: '文章版本快照、Diff 与局部回滚' },
{ label: '评论', url: /\/comments$/, text: '评论审核队列' },
{ label: '友链', url: /\/friend-links$/, text: '友链申请队列' },
{ label: '评测', url: /\/reviews$/, text: '评测内容库' },
{ label: '媒体库', url: /\/media$/, text: '对象存储媒体管理' },
{ label: '订阅', url: /\/subscriptions$/, text: '订阅中心 / 异步投递 / 汇总简报' },
{ label: 'Workers', url: /\/workers$/, text: '异步 Worker 控制台' },
{ label: '审计', url: /\/audit$/, text: '后台操作审计日志' },
{ label: '设置', url: /\/settings$/, text: '品牌、资料与 AI 控制' },
]
for (const route of routes) {
await page.getByRole('link', { name: route.label }).click()
await expect(page).toHaveURL(route.url)
await expect(page.locator('main')).toContainText(route.text)
}
})
test('后台 dashboard worker 健康卡片可跳转到带筛选的 workers', async ({ page }) => {
await loginAdmin(page)
await expect(page.locator('main')).toContainText('Worker 活动')
await page.getByTestId('dashboard-worker-card-failed').click()
await expect(page).toHaveURL(/\/workers\?status=failed$/)
await expect(page.locator('main')).toContainText('异步 Worker 控制台')
await expect(page.locator('main')).toContainText('failed')
})
test('后台可以审核评论和友链,并更新站点设置', async ({ page }) => {
await loginAdmin(page)
await page.getByRole('link', { name: '评论' }).click()
await expect(page.locator('main')).toContainText('Carol')
await page.getByRole('button', { name: '通过' }).first().click()
await page.getByRole('link', { name: '友链' }).click()
await expect(page.locator('main')).toContainText('Pending Link Review')
await page.getByRole('button', { name: '通过' }).first().click()
await page.getByRole('link', { name: '设置' }).click()
await expect(page.locator('main')).toContainText('InitCool')
await expect(page.getByTestId('site-settings-save')).toBeVisible()
})
test('后台可完成分类与标签的创建、更新、删除', async ({ page, request }) => {
await loginAdmin(page)
await page.getByRole('link', { name: '分类' }).click()
await page.getByTestId('category-name-input').fill('Playwright 深回归分类')
await page.getByTestId('category-slug-input').fill('playwright-deep-category')
await page.getByPlaceholder('介绍这个分类主要收录哪些内容。').fill('用于后台深度回归的分类。')
await page.getByTestId('category-save').click()
await expect(page.getByTestId('category-item-playwright-deep-category')).toBeVisible()
await page.getByTestId('category-item-playwright-deep-category').click()
await page.getByPlaceholder('前端工程专题 - Termi').fill('Playwright 深回归分类 SEO')
await page.getByTestId('category-save').click()
let state = await getDebugState(request)
expect(
state.categories.some(
(item: { slug: string; seo_title: string }) =>
item.slug === 'playwright-deep-category' &&
item.seo_title === 'Playwright 深回归分类 SEO',
),
).toBeTruthy()
acceptNextDialog(page)
await page.getByTestId('category-delete').click()
await expect(page.getByTestId('category-item-playwright-deep-category')).toHaveCount(0)
await page.getByRole('link', { name: '标签' }).click()
await page.getByTestId('tag-name-input').fill('Playwright 深回归标签')
await page.getByTestId('tag-slug-input').fill('playwright-deep-tag')
await page.getByPlaceholder('介绍这个标签常见主题、适合谁看。').fill('用于后台深度回归的标签。')
await page.getByTestId('tag-save').click()
await expect(page.getByTestId('tag-item-playwright-deep-tag')).toBeVisible()
await page.getByTestId('tag-item-playwright-deep-tag').click()
await page.getByPlaceholder('Astro 相关文章 - Termi').fill('Playwright 深回归标签 SEO')
await page.getByTestId('tag-save').click()
state = await getDebugState(request)
expect(
state.tags.some(
(item: { slug: string; seo_title: string }) =>
item.slug === 'playwright-deep-tag' && item.seo_title === 'Playwright 深回归标签 SEO',
),
).toBeTruthy()
acceptNextDialog(page)
await page.getByTestId('tag-delete').click()
await expect(page.getByTestId('tag-item-playwright-deep-tag')).toHaveCount(0)
})
test('后台可完成订阅 CRUD、测试投递与 digest 入队', async ({ page, request }) => {
await loginAdmin(page)
await page.getByRole('link', { name: '订阅' }).click()
await page.getByPlaceholder('name@example.com').fill('deep-regression@example.com')
await page.getByPlaceholder('例如 站长邮箱 / Discord 运维群').fill('Deep Regression')
await page.getByTestId('subscriptions-save').click()
const row = page
.locator('[data-testid^="subscription-row-"]')
.filter({ hasText: 'deep-regression@example.com' })
await expect(row).toBeVisible()
await row.getByTestId(/subscription-edit-/).click()
await page.getByPlaceholder('例如 站长邮箱 / Discord 运维群').fill('Deep Regression Updated')
await page.getByTestId('subscriptions-save').click()
await expect(row).toContainText('Deep Regression Updated')
await row.getByTestId(/subscription-test-/).click()
await expect(page.getByTestId('subscriptions-last-job')).toBeVisible()
await page.getByTestId('subscriptions-last-job').click()
await expect(page).toHaveURL(/\/workers\?job=\d+$/)
await expect(page.locator('main')).toContainText('subscription.test')
await page.getByRole('link', { name: '订阅' }).click()
await page.getByTestId('subscriptions-send-weekly').click()
await expect(page.getByTestId('subscriptions-last-job')).toBeVisible()
await page.getByTestId('subscriptions-send-monthly').click()
await expect(page.getByTestId(/^subscription-delivery-job-/).first()).toBeVisible()
await page.getByTestId(/^subscription-delivery-job-/).first().click()
await expect(page).toHaveURL(/\/workers\?job=\d+$/)
await expect(page.locator('main')).toContainText('worker.notification_delivery')
await page.getByRole('link', { name: '订阅' }).click()
let state = await getDebugState(request)
expect(
state.deliveries.some((item: { event_type: string; target: string }) =>
item.event_type === 'subscription.test' && item.target === 'deep-regression@example.com'),
).toBeTruthy()
expect(state.deliveries.some((item: { event_type: string }) => item.event_type === 'digest.weekly')).toBeTruthy()
expect(state.deliveries.some((item: { event_type: string }) => item.event_type === 'digest.monthly')).toBeTruthy()
expect(
state.worker_jobs.some((item: { worker_name: string }) => item.worker_name === 'worker.notification_delivery'),
).toBeTruthy()
await row.getByTestId(/subscription-delete-/).click()
await expect(row).toHaveCount(0)
state = await getDebugState(request)
expect(
state.subscriptions.some((item: { target: string }) => item.target === 'deep-regression@example.com'),
).toBeFalsy()
})
test('后台可查看 worker 控制台并执行 digest / retry / job 重跑', async ({ page, request }) => {
await loginAdmin(page)
await page.getByRole('link', { name: 'Workers' }).click()
await page.getByTestId('workers-run-weekly').click()
await expect(page.locator('main')).toContainText('task.send_weekly_digest')
await page.getByTestId('workers-retry-job').click()
let state = await getDebugState(request)
expect(
state.worker_jobs.some((item: { worker_name: string }) => item.worker_name === 'task.send_weekly_digest'),
).toBeTruthy()
expect(
state.worker_jobs.some(
(item: { worker_name: string; parent_job_id: number | null }) =>
item.worker_name === 'task.send_weekly_digest' && item.parent_job_id !== null,
),
).toBeTruthy()
await page.getByTestId('workers-run-retry').click()
state = await getDebugState(request)
expect(
state.worker_jobs.some((item: { worker_name: string }) => item.worker_name === 'task.retry_deliveries'),
).toBeTruthy()
})
test('后台可完成文章创建、保存、版本恢复与删除', async ({ page, request }) => {
await loginAdmin(page)
await page.getByRole('link', { name: '文章' }).click()
await page.getByTestId('posts-open-create').click()
await page.getByTestId('post-create-title').fill('Playwright 深回归文章')
await page.getByTestId('post-create-slug').fill('playwright-deep-post')
await page.getByTestId('post-create-submit').click()
await expect(page).toHaveURL(/\/posts\/playwright-deep-post$/)
await expect(page.getByTestId('post-editor-title')).toHaveValue('Playwright 深回归文章')
await page.getByTestId('post-editor-title').fill('Playwright 深回归文章(已更新)')
await page.getByTestId('post-editor-save').click()
await expect(page.getByTestId('post-editor-title')).toHaveValue('Playwright 深回归文章(已更新)')
let state = await getDebugState(request)
expect(
state.posts.some(
(item: { slug: string; title: string }) =>
item.slug === 'playwright-deep-post' && item.title === 'Playwright 深回归文章(已更新)',
),
).toBeTruthy()
await page.getByTestId('post-editor-close').click()
state = await getDebugState(request)
const createRevision = state.post_revisions.find(
(item: { id: number; post_slug: string; operation: string }) =>
item.post_slug === 'playwright-deep-post' && item.operation === 'create',
)
expect(createRevision).toBeTruthy()
await page.getByRole('link', { name: '版本' }).click()
await page.getByTestId('revisions-slug-filter').fill('playwright-deep-post')
await page.getByTestId(`revision-open-${createRevision.id}`).click()
await page.getByTestId('revision-restore-full').click()
await page.getByRole('link', { name: '文章' }).click()
await page.getByTestId('post-item-playwright-deep-post').click()
await expect(page.getByTestId('post-editor-title')).toHaveValue('Playwright 深回归文章')
acceptNextDialog(page)
await page.getByTestId('post-editor-delete').click()
await expect(page).toHaveURL(/\/posts$/)
state = await getDebugState(request)
expect(state.posts.some((item: { slug: string }) => item.slug === 'playwright-deep-post')).toBeFalsy()
})
test('后台可完成媒体库上传/元数据/替换/删除,并执行设置页关键动作', async ({ page, request }) => {
await loginAdmin(page)
await page.getByRole('link', { name: '媒体库' }).click()
await page.getByTestId('media-remote-url').fill(`${MOCK_BASE_URL}/media-files/remote-playwright.svg`)
await page.getByTestId('media-remote-title').fill('Remote Playwright Cover')
await page.getByTestId('media-remote-download').click()
await expect(page.getByTestId('media-last-remote-job')).toBeVisible()
await page.getByTestId('media-last-remote-job').click()
await expect(page).toHaveURL(/\/workers\?job=\d+$/)
await expect(page.locator('main')).toContainText('worker.download_media')
await page.getByRole('link', { name: '媒体库' }).click()
await page.getByTestId('media-upload-input').setInputFiles([
buildSvgPayload('deep-regression-cover.svg', 'deep-upload'),
])
await page.getByTestId('media-upload').click()
await expect(page.getByTestId('media-item-0')).toContainText('deep-regression-cover.svg')
await page.getByTestId('media-edit-0').click()
await page.getByPlaceholder('文章封面 / 站点横幅').fill('Deep Regression Cover')
await page.getByPlaceholder('夜色下的终端风格博客封面').fill('Deep Regression Alt')
await page.getByPlaceholder('cover, astro, terminal').fill('playwright, regression')
await page.getByTestId('media-save-metadata').click()
let state = await getDebugState(request)
expect(
state.media.some(
(item: { title: string; key: string }) =>
item.title === 'Remote Playwright Cover' &&
String(item.key || '').includes('post-covers/'),
),
).toBeTruthy()
expect(
state.media.some(
(item: { title: string; alt_text: string; tags: string[] }) =>
item.title === 'Deep Regression Cover' &&
item.alt_text === 'Deep Regression Alt' &&
item.tags.includes('playwright'),
),
).toBeTruthy()
await page.getByTestId('media-replace-input-0').setInputFiles([
buildSvgPayload('deep-regression-cover.svg', 'deep-replaced'),
])
acceptNextDialog(page)
await page.getByTestId('media-delete-0').click()
state = await getDebugState(request)
expect(state.media.some((item: { title: string }) => item.title === 'Deep Regression Cover')).toBeFalsy()
await page.getByRole('link', { name: '设置' }).click()
await page.getByTestId('site-settings-site-name').fill('InitCool Deep Regression')
await page.getByTestId('site-settings-popup-title').fill('订阅深回归')
await page
.locator('label', { hasText: '开启文章页微信扫码分享' })
.locator('input[type="checkbox"]')
.check()
await page.getByTestId('site-settings-save').click()
await page.getByTestId('site-settings-reindex').click()
await page.getByTestId('site-settings-test-provider').click()
await page.getByTestId('site-settings-test-image-provider').click()
await page.getByTestId('site-settings-test-storage').click()
state = await getDebugState(request)
expect(state.site_settings.site_name).toBe('InitCool Deep Regression')
expect(state.site_settings.subscription_popup_title).toBe('订阅深回归')
expect(state.site_settings.seo_wechat_share_qr_enabled).toBe(true)
expect(state.site_settings.ai_chunks_count).toBeGreaterThan(128)
})
test('后台可完成评测 CRUD、AI 润色,以及评论画像/黑名单管理', async ({ page, request }) => {
await loginAdmin(page)
await page.getByRole('link', { name: '评测' }).click()
await page.getByTestId('review-title').fill('Playwright 深评测')
await page.getByTestId('review-date').fill('2026-04-01')
await page.getByTestId('review-description').fill('这是一段用于深度回归的评测简介。')
await page.getByTestId('review-save').click()
await expect(page.locator('main')).toContainText('Playwright 深评测')
await page.getByTestId('review-ai-polish').click()
await expect(page.getByText('AI 点评润色对比')).toBeVisible()
await page.getByTestId('review-ai-adopt').click()
await page.getByTestId('review-save').click()
let state = await getDebugState(request)
expect(
state.reviews.some((item: { title: string }) => item.title === 'Playwright 深评测'),
).toBeTruthy()
acceptNextDialog(page)
await page.getByTestId('review-delete').click()
state = await getDebugState(request)
expect(
state.reviews.some((item: { title: string }) => item.title === 'Playwright 深评测'),
).toBeFalsy()
await page.getByRole('link', { name: '评论' }).click()
await page.getByRole('button', { name: 'AI 分析' }).click()
await expect(page.locator('main')).toContainText('建议保持观察')
await page.getByPlaceholder('输入要封禁的值').fill('203.0.113.55')
await page.getByPlaceholder('原因(可选)').first().fill('playwright deep regression')
await page.getByTestId('comment-blacklist-add').click()
state = await getDebugState(request)
const createdRule = state.comment_blacklist.find(
(item: { id: number; matcher_value: string }) => item.matcher_value === '203.0.113.55',
)
expect(createdRule).toBeTruthy()
await page.getByTestId(`blacklist-toggle-${createdRule.id}`).click()
await page.getByTestId(`blacklist-toggle-${createdRule.id}`).click()
acceptNextDialog(page)
await page.getByTestId(`blacklist-delete-${createdRule.id}`).click()
state = await getDebugState(request)
expect(
state.comment_blacklist.some((item: { matcher_value: string }) => item.matcher_value === '203.0.113.55'),
).toBeFalsy()
})