import type { Job } from 'bullmq' import { beforeEach, describe, expect, it, vi } from 'vitest' import { TASK_TYPE, type TaskJobData } from '@/lib/task/types' const prismaMock = vi.hoisted(() => ({ novelPromotionCharacter: { findFirst: vi.fn(), findMany: vi.fn(), update: vi.fn(async () => ({})), }, characterAppearance: { create: vi.fn(async () => ({})), }, })) const llmMock = vi.hoisted(() => ({ chatCompletion: vi.fn(async () => ({ id: 'completion-1' })), getCompletionContent: vi.fn(), })) const helperMock = vi.hoisted(() => ({ resolveProjectModel: vi.fn(async () => ({ id: 'project-1', novelPromotionData: { id: 'np-project-1', analysisModel: 'llm::analysis-1', }, })), })) const workerMock = vi.hoisted(() => ({ reportTaskProgress: vi.fn(async () => undefined), assertTaskActive: vi.fn(async () => undefined), })) vi.mock('@/lib/prisma', () => ({ prisma: prismaMock })) vi.mock('@/lib/llm-client', () => llmMock) vi.mock('@/types/character-profile', () => ({ validateProfileData: vi.fn(() => true), stringifyProfileData: vi.fn((value: unknown) => JSON.stringify(value)), })) vi.mock('@/lib/llm-observe/internal-stream-context', () => ({ withInternalLLMStreamCallbacks: vi.fn(async (_callbacks: unknown, fn: () => Promise) => await fn()), })) vi.mock('@/lib/workers/shared', () => ({ reportTaskProgress: workerMock.reportTaskProgress })) vi.mock('@/lib/workers/utils', () => ({ assertTaskActive: workerMock.assertTaskActive })) vi.mock('@/lib/workers/handlers/llm-stream', () => ({ createWorkerLLMStreamContext: vi.fn(() => ({ streamRunId: 'run-1', nextSeqByStepLane: {} })), createWorkerLLMStreamCallbacks: vi.fn(() => ({ onStage: vi.fn(), onChunk: vi.fn(), onComplete: vi.fn(), onError: vi.fn(), flush: vi.fn(async () => undefined), })), })) vi.mock('@/lib/workers/handlers/character-profile-helpers', async () => { const actual = await vi.importActual( '@/lib/workers/handlers/character-profile-helpers', ) return { ...actual, resolveProjectModel: helperMock.resolveProjectModel, } }) vi.mock('@/lib/prompt-i18n', () => ({ PROMPT_IDS: { NP_AGENT_CHARACTER_VISUAL: 'np_agent_character_visual' }, buildPrompt: vi.fn(() => 'character-visual-prompt'), })) import { handleCharacterProfileTask } from '@/lib/workers/handlers/character-profile' function buildJob(type: TaskJobData['type'], payload: Record): Job { return { data: { taskId: 'task-character-profile-1', type, locale: 'zh', projectId: 'project-1', episodeId: null, targetType: 'NovelPromotionCharacter', targetId: 'character-1', payload, userId: 'user-1', }, } as unknown as Job } describe('worker character-profile behavior', () => { beforeEach(() => { vi.clearAllMocks() llmMock.getCompletionContent.mockReturnValue( JSON.stringify({ characters: [ { appearances: [ { change_reason: '默认形象', descriptions: ['黑发,冷静,风衣'], }, ], }, ], }), ) prismaMock.novelPromotionCharacter.findFirst.mockImplementation(async (args: { where: { id: string } }) => ({ id: args.where.id, name: args.where.id === 'character-2' ? 'Villain' : 'Hero', profileData: JSON.stringify({ archetype: 'lead' }), profileConfirmed: false, novelPromotionProjectId: 'np-project-1', })) prismaMock.novelPromotionCharacter.findMany.mockResolvedValue([ { id: 'character-1', name: 'Hero', profileData: JSON.stringify({ archetype: 'lead' }), profileConfirmed: false, }, { id: 'character-2', name: 'Villain', profileData: JSON.stringify({ archetype: 'antagonist' }), profileConfirmed: false, }, ]) }) it('unsupported task type -> explicit error', async () => { const job = buildJob(TASK_TYPE.AI_CREATE_CHARACTER, {}) await expect(handleCharacterProfileTask(job)).rejects.toThrow('Unsupported character profile task type') }) it('confirm profile success -> creates appearance and marks profileConfirmed', async () => { const job = buildJob(TASK_TYPE.CHARACTER_PROFILE_CONFIRM, { characterId: 'character-1' }) const result = await handleCharacterProfileTask(job) expect(prismaMock.characterAppearance.create).toHaveBeenCalledWith({ data: expect.objectContaining({ characterId: 'character-1', appearanceIndex: 0, changeReason: '默认形象', description: '黑发,冷静,风衣', }), }) expect(prismaMock.novelPromotionCharacter.update).toHaveBeenCalledWith({ where: { id: 'character-1' }, data: { profileConfirmed: true }, }) expect(result).toEqual(expect.objectContaining({ success: true, character: expect.objectContaining({ id: 'character-1', profileConfirmed: true, }), })) }) it('batch confirm -> loops through all unconfirmed characters and returns count', async () => { const job = buildJob(TASK_TYPE.CHARACTER_PROFILE_BATCH_CONFIRM, {}) const result = await handleCharacterProfileTask(job) expect(result).toEqual({ success: true, count: 2, }) expect(prismaMock.characterAppearance.create).toHaveBeenCalledTimes(2) }) })