Files
waooplus/tests/integration/api/specific/novel-promotion-project-art-style-validation.test.ts
saturn 9aff44e37a refactor: analysis workflow architecture
fix: NEXTAUTH_URL

fix: prevent project model edits from affecting default model
2026-03-16 21:48:57 +08:00

123 lines
4.1 KiB
TypeScript

import { beforeEach, describe, expect, it, vi } from 'vitest'
import { buildMockRequest } from '../../../helpers/request'
const authMock = vi.hoisted(() => ({
requireProjectAuthLight: vi.fn(async () => ({
session: { user: { id: 'user-1', name: 'User 1' } },
project: { id: 'project-1', userId: 'user-1', mode: 'novel-promotion', name: 'Project 1' },
})),
isErrorResponse: vi.fn((value: unknown) => value instanceof Response),
}))
const prismaMock = vi.hoisted(() => ({
novelPromotionProject: {
findUnique: vi.fn(async () => ({
analysisModel: 'llm::analysis',
characterModel: 'img::character',
locationModel: 'img::location',
storyboardModel: 'img::storyboard',
editModel: 'img::edit',
videoModel: 'video::model',
audioModel: 'audio::model',
})),
update: vi.fn(async () => ({
id: 'np-1',
artStyle: 'realistic',
})),
},
userPreference: {
upsert: vi.fn(async () => ({ userId: 'user-1', artStyle: 'realistic' })),
},
}))
const mediaAttachMock = vi.hoisted(() => ({
attachMediaFieldsToProject: vi.fn(async (value: unknown) => value),
}))
const logMock = vi.hoisted(() => ({
logProjectAction: vi.fn(),
}))
const modelConfigContractMock = vi.hoisted(() => ({
parseModelKeyStrict: vi.fn(() => ({ provider: 'mock', modelId: 'mock-model' })),
}))
const capabilityLookupMock = vi.hoisted(() => ({
resolveBuiltinModelContext: vi.fn(() => null),
getCapabilityOptionFields: vi.fn(() => ({})),
validateCapabilitySelectionsPayload: vi.fn(() => []),
}))
vi.mock('@/lib/api-auth', () => authMock)
vi.mock('@/lib/prisma', () => ({ prisma: prismaMock }))
vi.mock('@/lib/media/attach', () => mediaAttachMock)
vi.mock('@/lib/logging/semantic', () => logMock)
vi.mock('@/lib/model-config-contract', () => modelConfigContractMock)
vi.mock('@/lib/model-capabilities/lookup', () => capabilityLookupMock)
describe('api specific - novel promotion project art style validation', () => {
beforeEach(() => {
vi.clearAllMocks()
})
it('accepts valid artStyle and keeps user preference unchanged', async () => {
const mod = await import('@/app/api/novel-promotion/[projectId]/route')
const req = buildMockRequest({
path: '/api/novel-promotion/project-1',
method: 'PATCH',
body: {
artStyle: ' realistic ',
},
})
const res = await mod.PATCH(req, { params: Promise.resolve({ projectId: 'project-1' }) })
expect(res.status).toBe(200)
expect(prismaMock.novelPromotionProject.update).toHaveBeenCalledWith(
expect.objectContaining({
data: expect.objectContaining({ artStyle: 'realistic' }),
}),
)
expect(prismaMock.userPreference.upsert).not.toHaveBeenCalled()
})
it('rejects invalid artStyle with invalid params', async () => {
const mod = await import('@/app/api/novel-promotion/[projectId]/route')
const req = buildMockRequest({
path: '/api/novel-promotion/project-1',
method: 'PATCH',
body: {
artStyle: 'anime',
},
})
const res = await mod.PATCH(req, { params: Promise.resolve({ projectId: 'project-1' }) })
const body = await res.json()
expect(res.status).toBe(400)
expect(body.error.code).toBe('INVALID_PARAMS')
expect(prismaMock.novelPromotionProject.update).not.toHaveBeenCalled()
expect(prismaMock.userPreference.upsert).not.toHaveBeenCalled()
})
it('accepts audioModel and keeps user preference unchanged', async () => {
const mod = await import('@/app/api/novel-promotion/[projectId]/route')
const req = buildMockRequest({
path: '/api/novel-promotion/project-1',
method: 'PATCH',
body: {
audioModel: 'bailian::qwen3-tts-vd-2026-01-26',
},
})
const res = await mod.PATCH(req, { params: Promise.resolve({ projectId: 'project-1' }) })
expect(res.status).toBe(200)
expect(prismaMock.novelPromotionProject.update).toHaveBeenCalledWith(
expect.objectContaining({
data: expect.objectContaining({
audioModel: 'bailian::qwen3-tts-vd-2026-01-26',
}),
}),
)
expect(prismaMock.userPreference.upsert).not.toHaveBeenCalled()
})
})