Files
aurak/web/services/questionBankService.ts
T
Developer 649844a657 feat: 添加复查功能和批量审核操作
- 复查功能: PUT /assessment/:id/review
  * 支持调整最终总分
  * 记录复查历史(reviewHistory)
  * 保存原始分数(originalScore)
  * 保留复查人、复查时间、复查意见

- 批量审核: POST /question-banks/:bankId/items/batch-review
  * 支持批量通过/拒绝题目
  * 可添加审核意见

- AssessmentSession实体: 添加复查相关字段
2026-05-13 23:06:40 +08:00

162 lines
6.0 KiB
TypeScript

import { apiClient } from './apiClient';
export interface QuestionBank {
id: string;
name: string;
description?: string;
status: 'DRAFT' | 'PENDING_REVIEW' | 'PUBLISHED' | 'REJECTED';
templateId?: string | null;
createdAt: string;
updatedAt: string;
}
export interface QuestionBankItem {
id: string;
bankId: string;
questionText: string;
questionType: 'SHORT_ANSWER' | 'MULTIPLE_CHOICE' | 'TRUE_FALSE';
options?: string[] | null;
correctAnswer?: string | null;
keyPoints: string[];
difficulty: 'STANDARD' | 'ADVANCED' | 'SPECIALIST';
dimension: 'PROMPT' | 'LLM' | 'IDE' | 'DEV_PATTERN' | 'WORK_CAPABILITY';
basis?: string | null;
status: 'PENDING_REVIEW' | 'PUBLISHED';
createdAt: string;
}
export interface CreateQuestionBankDto {
name: string;
description?: string;
templateId?: string;
}
export interface CreateQuestionBankItemDto {
questionText: string;
questionType: 'SHORT_ANSWER' | 'MULTIPLE_CHOICE' | 'TRUE_FALSE';
options?: string[];
correctAnswer?: string;
keyPoints: string[];
difficulty: 'STANDARD' | 'ADVANCED' | 'SPECIALIST';
dimension: 'PROMPT' | 'LLM' | 'IDE' | 'DEV_PATTERN' | 'WORK_CAPABILITY';
}
export const questionBankService = {
async getBanks(): Promise<QuestionBank[]> {
const response = await apiClient.request('/question-banks', {});
if (!response.ok) throw new Error('Failed to fetch question banks');
const data = await response.json();
return Array.isArray(data) ? data : data.data || [];
},
async getBank(id: string): Promise<QuestionBank> {
const response = await apiClient.request(`/question-banks/${id}`, {});
if (!response.ok) throw new Error('Failed to fetch question bank');
return await response.json();
},
async createBank(data: CreateQuestionBankDto): Promise<QuestionBank> {
const response = await apiClient.post('/question-banks', data);
if (!response.ok) throw new Error('Failed to create question bank');
return response.data;
},
async updateBank(id: string, data: Partial<CreateQuestionBankDto>): Promise<QuestionBank> {
const response = await apiClient.request(`/question-banks/${id}`, {
method: 'PUT',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify(data),
});
if (!response.ok) throw new Error('Failed to update question bank');
return await response.json();
},
async deleteBank(id: string): Promise<void> {
const response = await apiClient.request(`/question-banks/${id}`, { method: 'DELETE' });
if (!response.ok) throw new Error('Failed to delete question bank');
},
async submitForReview(id: string): Promise<QuestionBank> {
const response = await apiClient.request(`/question-banks/${id}/submit`, { method: 'PUT' });
if (!response.ok) throw new Error('Failed to submit for review');
return await response.json();
},
async approveBank(id: string): Promise<QuestionBank> {
const response = await apiClient.request(`/question-banks/${id}/review`, {
method: 'PUT',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ approved: true }),
});
if (!response.ok) throw new Error('Failed to approve');
return await response.json();
},
async rejectBank(id: string): Promise<QuestionBank> {
const response = await apiClient.request(`/question-banks/${id}/review`, {
method: 'PUT',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ approved: false }),
});
if (!response.ok) throw new Error('Failed to reject');
return await response.json();
},
async publishBank(id: string): Promise<QuestionBank> {
const response = await apiClient.request(`/question-banks/${id}/publish`, { method: 'PUT' });
if (!response.ok) throw new Error('Failed to publish');
return await response.json();
},
async getBankItems(bankId: string): Promise<QuestionBankItem[]> {
const response = await apiClient.request(`/question-banks/${bankId}`, {});
if (!response.ok) throw new Error('Failed to fetch items');
const data = await response.json();
return data.items || [];
},
async createItem(bankId: string, data: CreateQuestionBankItemDto): Promise<QuestionBankItem> {
const response = await apiClient.request(`/question-banks/${bankId}/items`, {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify(data),
});
if (!response.ok) throw new Error('Failed to create item');
return await response.json();
},
async updateItem(bankId: string, itemId: string, data: Partial<CreateQuestionBankItemDto>): Promise<QuestionBankItem> {
const response = await apiClient.request(`/question-banks/${bankId}/items/${itemId}`, {
method: 'PUT',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify(data),
});
if (!response.ok) throw new Error('Failed to update item');
return await response.json();
},
async deleteItem(bankId: string, itemId: string): Promise<void> {
const response = await apiClient.request(`/question-banks/${bankId}/items/${itemId}`, { method: 'DELETE' });
if (!response.ok) throw new Error('Failed to delete item');
},
async generateQuestions(bankId: string, count: number, knowledgeBaseContent: string): Promise<QuestionBankItem[]> {
const response = await apiClient.request(`/question-banks/${bankId}/generate`, {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ count, knowledgeBaseContent }),
});
if (!response.ok) throw new Error('Failed to generate questions');
return await response.json();
},
async batchReviewItems(bankId: string, itemIds: string[], approved: boolean, comment?: string): Promise<QuestionBankItem[]> {
const response = await apiClient.request(`/question-banks/${bankId}/items/batch-review`, {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ itemIds, approved, comment }),
});
if (!response.ok) throw new Error('Failed to batch review items');
return await response.json();
},
};