- 替换 QuickCaptureFlow 和 ArchiveFlow 为 UnifiedCaptureFlow 统一流程 - 新增 VLSession 封装 Qwen2.5-VL 模型进行图像文本推理 - 实现 AIRuntime 中 VL 模型的准备和分析功能 - 添加 VLPrompts 定义体检化验单识别的 JSON 输出模板 - 创建 CaptureReviewForm 提供 VL 解析结果的可编辑表单界面 - 集成 VisionKit 文档扫描器支持真机多页文档扫描 - 为模拟器实现 PhotosPicker 回退方案选择已有照片 - 在 RootView 中统一使用 UnifiedCaptureFlow 处理快速和归档流程 - 添加 CustomMetricEditor 支持自定义监测指标的创建编辑删除 - 扩展 KangkangApp 模型配置以支持新数据类型 - 实现档案列表中症状结束功能通过时间线行点击触发
163 lines
5.0 KiB
Swift
163 lines
5.0 KiB
Swift
import Testing
|
|
import SwiftData
|
|
import Foundation
|
|
@testable import 康康
|
|
|
|
struct ModelsSchemaTests {
|
|
|
|
private func makeContainer() throws -> ModelContainer {
|
|
let schema = Schema([
|
|
Indicator.self,
|
|
Report.self,
|
|
DiaryEntry.self,
|
|
Asset.self,
|
|
ChatTurn.self,
|
|
Symptom.self,
|
|
UserProfile.self,
|
|
MetricReminder.self,
|
|
CustomMonitorMetric.self,
|
|
])
|
|
let config = ModelConfiguration(schema: schema, isStoredInMemoryOnly: true)
|
|
return try ModelContainer(for: schema, configurations: [config])
|
|
}
|
|
|
|
@Test func insertIndicatorWithReportRelationship() throws {
|
|
let container = try makeContainer()
|
|
let ctx = ModelContext(container)
|
|
|
|
let report = Report(title: "春检", type: .checkup, reportDate: .now)
|
|
let indicator = Indicator(
|
|
name: "ALT",
|
|
value: "32",
|
|
unit: "U/L",
|
|
range: "9-50",
|
|
status: .normal,
|
|
report: report
|
|
)
|
|
ctx.insert(report)
|
|
ctx.insert(indicator)
|
|
try ctx.save()
|
|
|
|
#expect(report.indicators.count == 1)
|
|
#expect(indicator.report?.title == "春检")
|
|
}
|
|
|
|
@Test func cascadeDeleteReportRemovesIndicators() throws {
|
|
let container = try makeContainer()
|
|
let ctx = ModelContext(container)
|
|
|
|
let report = Report(title: "春检", type: .checkup, reportDate: .now)
|
|
let indicator = Indicator(
|
|
name: "ALT", value: "32", unit: "U/L", range: "9-50",
|
|
status: .normal, report: report
|
|
)
|
|
ctx.insert(report)
|
|
ctx.insert(indicator)
|
|
try ctx.save()
|
|
|
|
ctx.delete(report)
|
|
try ctx.save()
|
|
|
|
let remaining = try ctx.fetch(FetchDescriptor<Indicator>())
|
|
#expect(remaining.isEmpty)
|
|
}
|
|
|
|
@Test func ongoingSymptomQueryFiltersByEndedAt() throws {
|
|
let container = try makeContainer()
|
|
let ctx = ModelContext(container)
|
|
|
|
let active = Symptom(name: "头痛", startedAt: .now.addingTimeInterval(-3600))
|
|
let ended = Symptom(
|
|
name: "咳嗽",
|
|
startedAt: .now.addingTimeInterval(-7200),
|
|
endedAt: .now.addingTimeInterval(-1800)
|
|
)
|
|
ctx.insert(active)
|
|
ctx.insert(ended)
|
|
try ctx.save()
|
|
|
|
let predicate = #Predicate<Symptom> { $0.endedAt == nil }
|
|
let ongoing = try ctx.fetch(FetchDescriptor<Symptom>(predicate: predicate))
|
|
|
|
#expect(ongoing.count == 1)
|
|
#expect(ongoing.first?.name == "头痛")
|
|
#expect(active.isOngoing)
|
|
#expect(!ended.isOngoing)
|
|
#expect(active.duration >= 3600)
|
|
}
|
|
|
|
@Test func symptomSeverityClampedToRange() throws {
|
|
let high = Symptom(name: "腹痛", severity: 99)
|
|
let low = Symptom(name: "失眠", severity: -3)
|
|
#expect(high.severity == 5)
|
|
#expect(low.severity == 1)
|
|
}
|
|
|
|
@Test func chatTurnPersistsReferencedIDs() throws {
|
|
let container = try makeContainer()
|
|
let ctx = ModelContext(container)
|
|
|
|
let turn = ChatTurn(
|
|
question: "我的 LDL 怎么样?",
|
|
answer: "近 3 个月 LDL 偏高 [1]",
|
|
referencedIndicatorIDs: ["abc"],
|
|
referencedReportIDs: [],
|
|
decodeRate: 24.3
|
|
)
|
|
ctx.insert(turn)
|
|
try ctx.save()
|
|
|
|
let all = try ctx.fetch(FetchDescriptor<ChatTurn>())
|
|
#expect(all.count == 1)
|
|
#expect(all.first?.referencedIndicatorIDs == ["abc"])
|
|
}
|
|
|
|
@Test func indicatorSeriesKeyRoundtrip() throws {
|
|
let container = try makeContainer()
|
|
let ctx = ModelContext(container)
|
|
|
|
let bp = Indicator(
|
|
name: "收缩压",
|
|
value: "125",
|
|
unit: "mmHg",
|
|
range: "90-140",
|
|
status: .normal,
|
|
pinned: true,
|
|
seriesKey: "bp.systolic"
|
|
)
|
|
ctx.insert(bp)
|
|
try ctx.save()
|
|
|
|
let fetched = try #require(try ctx.fetch(FetchDescriptor<Indicator>()).first)
|
|
#expect(fetched.seriesKey == "bp.systolic")
|
|
#expect(fetched.pinned == true)
|
|
}
|
|
|
|
@Test func indicatorSeriesKeyDefaultsToNil() {
|
|
let i = Indicator(name: "ALT", value: "32", unit: "U/L", range: "9-50", status: .normal)
|
|
#expect(i.seriesKey == nil)
|
|
}
|
|
|
|
@Test func userProfileSchemaPersistsAcrossSave() throws {
|
|
let container = try makeContainer()
|
|
let ctx = ModelContext(container)
|
|
|
|
let p = UserProfile(
|
|
birthYear: 1985,
|
|
biologicalSexRaw: "male",
|
|
heightCM: 175,
|
|
bloodTypeRaw: "A",
|
|
chronicConditions: ["高血压"]
|
|
)
|
|
ctx.insert(p)
|
|
try ctx.save()
|
|
|
|
let fetched = try #require(try ctx.fetch(FetchDescriptor<UserProfile>()).first)
|
|
#expect(fetched.birthYear == 1985)
|
|
#expect(fetched.sex == .male)
|
|
#expect(fetched.heightCM == 175)
|
|
#expect(fetched.bloodTypeRaw == "A")
|
|
#expect(fetched.chronicConditions == ["高血压"])
|
|
}
|
|
}
|