Initial commit: 健康管家 AI 健康陪伴助手

- Backend: .NET 10 Minimal API + EF Core + PostgreSQL
- Frontend: Flutter + Riverpod + GoRouter + Dio
- AI: DeepSeek LLM + Qwen VLM (OpenAI-compatible)
- Auth: SMS + JWT (access/refresh tokens)
- Features: AI chat, health tracking, medication management, diet analysis, exercise plans, doctor consultations, report analysis
This commit is contained in:
MingNian
2026-06-02 11:11:29 +08:00
commit 14d7c30d3d
144 changed files with 11436 additions and 0 deletions

View File

@@ -0,0 +1,84 @@
import 'dart:async';
import 'dart:convert';
import 'package:dio/dio.dart';
import '../core/api_client.dart';
/// 跨平台 SSE 流处理(基于 Dio 流式响应,支持 Android/iOS/Web
class SseHandler {
/// 连接 SSE 端点,返回事件流
static Stream<Map<String, dynamic>> connect({
required String agentType,
required String message,
String? conversationId,
required String token,
}) {
final params = <String, String>{
'message': message,
'token': token,
};
if (conversationId != null) {
params['conversationId'] = conversationId;
}
final query = params.entries
.map((e) => '${e.key}=${Uri.encodeComponent(e.value)}')
.join('&');
final url = '$baseUrl/api/ai/$agentType/chat?$query';
final controller = StreamController<Map<String, dynamic>>();
_connect(controller, url);
return controller.stream;
}
static Future<void> _connect(
StreamController<Map<String, dynamic>> controller,
String url,
) async {
try {
final dio = Dio(BaseOptions(
connectTimeout: const Duration(seconds: 15),
receiveTimeout: const Duration(minutes: 5),
));
final response = await dio.get(
url,
options: Options(responseType: ResponseType.stream),
);
final stream = response.data.stream as Stream<List<int>>;
var buffer = '';
await for (final chunk in stream) {
if (controller.isClosed) break;
final text = utf8.decode(chunk, allowMalformed: true);
buffer += text;
// 按行解析 SSE 数据
while (buffer.contains('\n')) {
final newlineIdx = buffer.indexOf('\n');
var line = buffer.substring(0, newlineIdx).trim();
buffer = buffer.substring(newlineIdx + 1);
if (line.isEmpty || !line.startsWith('data: ')) continue;
final data = line.substring(6);
if (data == '[DONE]') {
controller.close();
return;
}
try {
controller.add(jsonDecode(data) as Map<String, dynamic>);
} catch (_) {
// 跳过无法解析的行
}
}
}
controller.close();
} catch (e) {
if (!controller.isClosed) {
controller.add({'action': 'error', 'message': e.toString()});
controller.close();
}
}
}
}