Files
AIclinicalresearch/backend/scripts/check-dc-tables.mjs
HaHafeng 5523ef36ea feat(admin): Complete Phase 3.5.1-3.5.4 Prompt Management System (83%)
Summary:
- Implement Prompt management infrastructure and core services
- Build admin portal frontend with light theme
- Integrate CodeMirror 6 editor for non-technical users

Phase 3.5.1: Infrastructure Setup
- Create capability_schema for Prompt storage
- Add prompt_templates and prompt_versions tables
- Add prompt:view/edit/debug/publish permissions
- Migrate RVW prompts to database (RVW_EDITORIAL, RVW_METHODOLOGY)

Phase 3.5.2: PromptService Core
- Implement gray preview logic (DRAFT for debuggers, ACTIVE for users)
- Module-level debug control (setDebugMode)
- Handlebars template rendering
- Variable extraction and validation (extractVariables, validateVariables)
- Three-level disaster recovery (database -> cache -> hardcoded fallback)

Phase 3.5.3: Management API
- 8 RESTful endpoints (/api/admin/prompts/*)
- Permission control (PROMPT_ENGINEER can edit, SUPER_ADMIN can publish)

Phase 3.5.4: Frontend Management UI
- Build admin portal architecture (AdminLayout, OrgLayout)
- Add route system (/admin/*, /org/*)
- Implement PromptListPage (filter, search, debug switch)
- Implement PromptEditor (CodeMirror 6 simplified for clinical users)
- Implement PromptEditorPage (edit, save, publish, test, version history)

Technical Details:
- Backend: 6 files, ~2044 lines (prompt.service.ts 596 lines)
- Frontend: 9 files, ~1735 lines (PromptEditorPage.tsx 399 lines)
- CodeMirror 6: Line numbers, auto-wrap, variable highlight, search, undo/redo
- Chinese-friendly: 15px font, 1.8 line-height, system fonts

Next Step: Phase 3.5.5 - Integrate RVW module with PromptService

Tested: Backend API tests passed (8/8), Frontend pending user testing
Status: Ready for Phase 3.5.5 RVW integration
2026-01-11 21:25:16 +08:00

251 lines
6.8 KiB
JavaScript
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
/**
* DC模块数据库表检查脚本使用Prisma
*
* 验证dc_schema和4个表是否已创建
*/
import { PrismaClient } from '@prisma/client';
const prisma = new PrismaClient();
async function checkDCTables() {
try {
console.log('');
console.log('============================================================');
console.log('[DC模块] 数据库表检查');
console.log('============================================================');
console.log('');
console.log('✅ Prisma连接初始化成功');
console.log('');
// 1. 检查dc_schema是否存在
console.log('📋 检查1: dc_schema是否存在');
const schemaResult = await prisma.$queryRawUnsafe(`
SELECT schema_name
FROM information_schema.schemata
WHERE schema_name = 'dc_schema'
`);
if (schemaResult.length === 0) {
console.log('❌ dc_schema 不存在!');
console.log('');
console.log('💡 解决方案:');
console.log(' cd backend');
console.log(' npx prisma db push');
console.log('');
await prisma.$disconnect();
process.exit(1);
}
console.log('✅ dc_schema 存在');
console.log('');
// 2. 检查4个表是否存在
console.log('📋 检查2: DC模块的4个表是否存在');
console.log('');
const tables = [
{ name: 'dc_health_checks', display: '健康检查表', model: 'dCHealthCheck' },
{ name: 'dc_templates', display: '预设模板表', model: 'dCTemplate' },
{ name: 'dc_extraction_tasks', display: '提取任务表', model: 'dCExtractionTask' },
{ name: 'dc_extraction_items', display: '提取明细表', model: 'dCExtractionItem' },
];
let allTablesExist = true;
const missingTables = [];
const tableCounts = {};
for (const table of tables) {
try {
const tableResult = await prisma.$queryRawUnsafe(`
SELECT table_name
FROM information_schema.tables
WHERE table_schema = 'dc_schema'
AND table_name = '${table.name}'
`);
if (tableResult.length > 0) {
// 获取行数
const countResult = await prisma.$queryRawUnsafe(`
SELECT COUNT(*) as count FROM dc_schema.${table.name}
`);
const count = Number(countResult[0].count);
tableCounts[table.name] = count;
console.log(`${table.display} (${table.name})`);
console.log(` 记录数: ${count}`);
} else {
console.log(`${table.display} (${table.name}): 不存在`);
allTablesExist = false;
missingTables.push(table.name);
}
} catch (error) {
console.log(`${table.display} (${table.name}): 检查失败 - ${error.message}`);
allTablesExist = false;
missingTables.push(table.name);
}
}
console.log('');
// 3. 检查dc_templates是否有预设数据
if (allTablesExist) {
console.log('📋 检查3: dc_templates预设模板是否存在');
const templateCount = tableCounts['dc_templates'];
if (templateCount === 0) {
console.log('⚠️ dc_templates表为空没有预设模板');
console.log('');
console.log('💡 需要启动后端服务初始化预设模板:');
console.log(' cd backend');
console.log(' npm run dev');
console.log(' 启动时会自动seed 3个预设模板');
} else {
console.log(`✅ dc_templates已有 ${templateCount} 个预设模板`);
// 列出模板
try {
const templates = await prisma.$queryRawUnsafe(`
SELECT disease_type, report_type, display_name
FROM dc_schema.dc_templates
ORDER BY created_at
`);
if (templates.length > 0) {
console.log('');
console.log(' 预设模板列表:');
templates.forEach((t, i) => {
console.log(` ${i + 1}. ${t.display_name} (${t.disease_type}/${t.report_type})`);
});
}
} catch (error) {
console.log(' ⚠️ 无法获取模板详情');
}
}
console.log('');
}
// 4. 总结
console.log('============================================================');
console.log('[总结]');
console.log('============================================================');
console.log('');
if (allTablesExist) {
console.log('🎉 恭喜DC模块数据库表已全部创建');
console.log('');
console.log('✅ dc_schema: 存在');
console.log('✅ 4个数据表: 全部存在');
console.log('');
console.log('📊 数据统计:');
console.log(` - dc_health_checks: ${tableCounts['dc_health_checks']}`);
console.log(` - dc_templates: ${tableCounts['dc_templates']}`);
console.log(` - dc_extraction_tasks: ${tableCounts['dc_extraction_tasks']}`);
console.log(` - dc_extraction_items: ${tableCounts['dc_extraction_items']}`);
console.log('');
console.log('📌 下一步:');
if (tableCounts['dc_templates'] === 0) {
console.log(' 1. ⚠️ 启动后端初始化预设模板npm run dev');
console.log(' 2. 然后可以开始前端开发!');
} else {
console.log(' ✅ 可以开始前端开发了!');
}
console.log('');
await prisma.$disconnect();
process.exit(0);
} else {
console.log('⚠️ DC模块数据库表未完全创建');
console.log('');
console.log('❌ 缺少以下表:');
missingTables.forEach(t => console.log(` - ${t}`));
console.log('');
console.log('💡 解决方案:');
console.log(' cd backend');
console.log(' npx prisma db push');
console.log('');
await prisma.$disconnect();
process.exit(1);
}
} catch (error) {
console.error('');
console.error('❌ 检查失败:', error.message);
console.error('');
if (error.message.includes('connect') || error.message.includes('ECONNREFUSED')) {
console.error('💡 数据库连接失败,请确认:');
console.error(' 1. PostgreSQL服务是否已启动');
console.error(' 2. DATABASE_URL环境变量是否正确');
console.error(' 当前: ' + (process.env.DATABASE_URL || '未设置'));
console.error('');
console.error(' 期望格式:');
console.error(' postgresql://postgres:postgres123@localhost:5432/ai_clinical_research');
}
console.error('');
console.error('详细错误:');
console.error(error);
console.error('');
await prisma.$disconnect().catch(() => {});
process.exit(1);
}
}
// 执行检查
checkDCTables();