DeepSeekService.cs 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798
  1. using System.Net.Http;
  2. using System.Net.Http.Headers;
  3. using System.Text.Json;
  4. using System.Text.Json.Serialization;
  5. using JsonSerializer = System.Text.Json.JsonSerializer;
  6. using System.IO;
  7. namespace OASystem.API.OAMethodLib.DeepSeekAPI
  8. {
  9. /// <summary>
  10. /// DeepSeek API 服务实现
  11. /// </summary>
  12. public class DeepSeekService : IDeepSeekService
  13. {
  14. private readonly HttpClient _httpClient;
  15. private readonly ILogger<DeepSeekService> _logger;
  16. private readonly IHostEnvironment _hostEnvironment;
  17. /// <summary>
  18. /// 配置文件
  19. /// </summary>
  20. private DeepSeek DeepSeek { get; set; }
  21. /// <summary>
  22. /// 构造函数
  23. /// </summary>
  24. public DeepSeekService(IHostEnvironment hostEnvironment, ILogger<DeepSeekService> logger, HttpClient httpClient)
  25. {
  26. _hostEnvironment = hostEnvironment;
  27. _httpClient = httpClient;
  28. _logger = logger;
  29. DeepSeek = AutofacIocManager.Instance.GetService<DeepSeek>();
  30. // 设置基础地址和认证头
  31. _httpClient.BaseAddress = new Uri(DeepSeek.BaseAddress ?? "https://api.deepseek.com/v1/");
  32. _httpClient.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", DeepSeek.ApiKey);
  33. _httpClient.DefaultRequestHeaders.Add("Accept", "application/json");
  34. }
  35. /// <summary>
  36. /// 测试API连通性
  37. /// </summary>
  38. /// <returns></returns>
  39. public async Task<bool> TestApiConnectivityAsync()
  40. {
  41. try
  42. {
  43. var response = await _httpClient.GetAsync("models");
  44. return response.IsSuccessStatusCode;
  45. }
  46. catch
  47. {
  48. return false;
  49. }
  50. }
  51. /// <summary>
  52. /// 检查可用端点
  53. /// </summary>
  54. /// <returns></returns>
  55. public async Task<List<string>> DiscoverAvailableEndpointsAsync()
  56. {
  57. var endpoints = new List<string>
  58. {
  59. "models",
  60. "chat/completions",
  61. "files",
  62. "uploads",
  63. "documents",
  64. "assistants"
  65. };
  66. var availableEndpoints = new List<string>();
  67. foreach (var endpoint in endpoints)
  68. {
  69. try
  70. {
  71. var response = await _httpClient.GetAsync(endpoint);
  72. if (response.IsSuccessStatusCode)
  73. {
  74. availableEndpoints.Add(endpoint);
  75. Console.WriteLine($"✅ 端点可用: {endpoint}");
  76. }
  77. else
  78. {
  79. Console.WriteLine($"❌ 端点不可用: {endpoint} - {response.StatusCode}");
  80. }
  81. }
  82. catch (Exception ex)
  83. {
  84. Console.WriteLine($"⚠️ 端点检查错误: {endpoint} - {ex.Message}");
  85. }
  86. }
  87. return availableEndpoints;
  88. }
  89. /// <summary>
  90. /// 上传文件到DeepSeek API
  91. /// </summary>
  92. public async Task<DeepSeekFileUploadResponse> UploadFileAsync(IFormFile file, string purpose = "assistants")
  93. {
  94. try
  95. {
  96. _logger.LogInformation("开始上传文件: {FileName}, 大小: {Size} bytes", file.FileName, file.Length);
  97. // 检查文件大小
  98. if (file.Length > 512 * 1024 * 1024) // 512MB限制
  99. {
  100. throw new Exception($"文件大小超过限制: {file.Length} bytes");
  101. }
  102. using var content = new MultipartFormDataContent();
  103. using var fileStream = file.OpenReadStream();
  104. var fileContent = new StreamContent(fileStream);
  105. fileContent.Headers.ContentType = new MediaTypeHeaderValue(GetContentType(file.FileName));
  106. content.Add(fileContent, "file", file.FileName);
  107. content.Add(new StringContent(purpose), "purpose");
  108. var response = await _httpClient.PostAsync("files", content);
  109. _logger.LogInformation("文件上传路径:{filePath}", response.RequestMessage.RequestUri);
  110. response.EnsureSuccessStatusCode();
  111. var responseContent = await response.Content.ReadAsStringAsync();
  112. var result = JsonSerializer.Deserialize<DeepSeekFileUploadResponse>(responseContent, new JsonSerializerOptions
  113. {
  114. PropertyNameCaseInsensitive = true
  115. });
  116. _logger.LogInformation("文件上传成功: {FileName}, FileId: {FileId}", file.FileName, result.Id);
  117. return result;
  118. }
  119. catch (Exception ex)
  120. {
  121. _logger.LogError(ex, "文件上传失败: {FileName}", file.FileName);
  122. throw;
  123. }
  124. }
  125. /// <summary>
  126. /// 批量上传文件
  127. /// </summary>
  128. public async Task<List<FileUploadResult>> UploadFilesAsync(List<IFormFile> files, string purpose = "assistants")
  129. {
  130. var results = new List<FileUploadResult>();
  131. foreach (var file in files)
  132. {
  133. var result = new FileUploadResult
  134. {
  135. FileName = file.FileName,
  136. FileSize = file.Length
  137. };
  138. try
  139. {
  140. var uploadResponse = await UploadFileAsync(file, purpose);
  141. result.FileId = uploadResponse.Id;
  142. result.Success = true;
  143. result.Status = uploadResponse.Status;
  144. result.Message = "上传成功";
  145. }
  146. catch (Exception ex)
  147. {
  148. result.Success = false;
  149. result.Message = $"上传失败: {ex.Message}";
  150. result.Status = "error";
  151. }
  152. results.Add(result);
  153. }
  154. return results;
  155. }
  156. /// <summary>
  157. /// 获取文件列表
  158. /// </summary>
  159. public async Task<DeepSeekFileListResponse> ListFilesAsync()
  160. {
  161. try
  162. {
  163. var response = await _httpClient.GetAsync("files");
  164. response.EnsureSuccessStatusCode();
  165. var responseContent = await response.Content.ReadAsStringAsync();
  166. return JsonSerializer.Deserialize<DeepSeekFileListResponse>(responseContent, new JsonSerializerOptions
  167. {
  168. PropertyNameCaseInsensitive = true
  169. });
  170. }
  171. catch (Exception ex)
  172. {
  173. _logger.LogError(ex, "获取文件列表失败");
  174. throw;
  175. }
  176. }
  177. /// <summary>
  178. /// 获取文件信息
  179. /// </summary>
  180. public async Task<DeepSeekFileUploadResponse> GetFileInfoAsync(string fileId)
  181. {
  182. try
  183. {
  184. var response = await _httpClient.GetAsync($"files/{fileId}");
  185. response.EnsureSuccessStatusCode();
  186. var responseContent = await response.Content.ReadAsStringAsync();
  187. return JsonSerializer.Deserialize<DeepSeekFileUploadResponse>(responseContent, new JsonSerializerOptions
  188. {
  189. PropertyNameCaseInsensitive = true
  190. });
  191. }
  192. catch (Exception ex)
  193. {
  194. _logger.LogError(ex, "获取文件信息失败: {FileId}", fileId);
  195. throw;
  196. }
  197. }
  198. /// <summary>
  199. /// 删除文件
  200. /// </summary>
  201. public async Task<bool> DeleteFileAsync(string fileId)
  202. {
  203. try
  204. {
  205. var response = await _httpClient.DeleteAsync($"files/{fileId}");
  206. response.EnsureSuccessStatusCode();
  207. _logger.LogInformation("文件删除成功: {FileId}", fileId);
  208. return true;
  209. }
  210. catch (Exception ex)
  211. {
  212. _logger.LogError(ex, "文件删除失败: {FileId}", fileId);
  213. return false;
  214. }
  215. }
  216. /// <summary>
  217. /// 使用已上传的文件进行聊天
  218. /// </summary>
  219. public async Task<ApiResponse> ChatWithFilesAsync(List<string> fileIds, string question, string model = "deepseek-chat", float temperature = 0.7f, int maxTokens = 4000)
  220. {
  221. try
  222. {
  223. // 等待所有文件处理完成
  224. var processedFiles = new List<DeepSeekFileUploadResponse>();
  225. foreach (var fileId in fileIds)
  226. {
  227. var fileInfo = await WaitForFileProcessingAsync(fileId);
  228. processedFiles.Add(fileInfo);
  229. }
  230. // 构建消息内容
  231. var messageContent = new List<object>
  232. {
  233. new { type = "text", text = question }
  234. };
  235. // 添加文件引用
  236. foreach (var file in processedFiles)
  237. {
  238. messageContent.Add(new
  239. {
  240. type = "file",
  241. file_id = file.Id
  242. });
  243. }
  244. var request = new DeepSeekChatWithFilesRequest
  245. {
  246. Model = model,
  247. Messages = new List<FileMessage>
  248. {
  249. new FileMessage
  250. {
  251. Role = "user",
  252. Content = messageContent
  253. }
  254. },
  255. Temperature = temperature,
  256. MaxTokens = maxTokens
  257. };
  258. var jsonContent = JsonSerializer.Serialize(request, new JsonSerializerOptions
  259. {
  260. PropertyNamingPolicy = JsonNamingPolicy.CamelCase
  261. });
  262. var httpContent = new StringContent(jsonContent, Encoding.UTF8, "application/json");
  263. var response = await _httpClient.PostAsync("chat/completions", httpContent);
  264. response.EnsureSuccessStatusCode();
  265. var responseContent = await response.Content.ReadAsStringAsync();
  266. var chatResponse = JsonSerializer.Deserialize<DeepSeekResponse>(responseContent, new JsonSerializerOptions
  267. {
  268. PropertyNameCaseInsensitive = true
  269. });
  270. return new ApiResponse
  271. {
  272. Success = true,
  273. Message = "聊天成功",
  274. Answer = chatResponse.Choices[0].Message.Content,
  275. TokensUsed = chatResponse.Usage.TotalTokens
  276. };
  277. }
  278. catch (Exception ex)
  279. {
  280. _logger.LogError(ex, "使用文件聊天失败");
  281. throw;
  282. }
  283. }
  284. /// <summary>
  285. /// 使用进行聊天
  286. /// </summary>
  287. /// <param name="question"></param>
  288. /// <param name="model"></param>
  289. /// <param name="stream"></param>
  290. /// <param name="temperature"></param>
  291. /// <param name="maxTokens"></param>
  292. /// <returns></returns>
  293. public async Task<ApiResponse> ChatAsync(string question, bool stream = false, string model = "deepseek-chat", float temperature = 0.7f, int maxTokens = 4000)
  294. {
  295. try
  296. {
  297. // 构建消息内容
  298. var messageContent = new List<object>
  299. {
  300. new { type = "text", text = question }
  301. };
  302. var request = new DeepSeekChatWithFilesRequest
  303. {
  304. Model = model,
  305. Messages = new List<FileMessage>
  306. {
  307. new FileMessage
  308. {
  309. Role = "user",
  310. Content = messageContent
  311. }
  312. },
  313. Stream = stream,
  314. Temperature = temperature,
  315. MaxTokens = maxTokens,
  316. TopP = 0.9M,
  317. FrequencyPenalty = 0.2M,
  318. PresencePenalty = 0.1M,
  319. };
  320. var jsonContent = JsonSerializer.Serialize(request, new JsonSerializerOptions
  321. {
  322. PropertyNamingPolicy = JsonNamingPolicy.CamelCase
  323. });
  324. var httpContent = new StringContent(jsonContent, Encoding.UTF8, "application/json");
  325. var response = await _httpClient.PostAsync("chat/completions", httpContent);
  326. response.EnsureSuccessStatusCode();
  327. var responseContent = await response.Content.ReadAsStringAsync();
  328. var chatResponse = JsonSerializer.Deserialize<DeepSeekResponse>(responseContent, new JsonSerializerOptions
  329. {
  330. PropertyNameCaseInsensitive = true
  331. });
  332. return new ApiResponse
  333. {
  334. Success = true,
  335. Message = "聊天成功",
  336. Answer = chatResponse.Choices[0].Message.Content,
  337. TokensUsed = chatResponse.Usage.TotalTokens
  338. };
  339. }
  340. catch (Exception ex)
  341. {
  342. _logger.LogError(ex, "使用文件聊天失败");
  343. throw;
  344. }
  345. }
  346. /// <summary>
  347. /// 流式 chat;思考与正文分 Phase 产出
  348. /// </summary>
  349. /// <param name="question">问题</param>
  350. /// <param name="model">模型名称</param>
  351. /// <param name="temperature">温度参数</param>
  352. /// <param name="maxTokens">最大token数</param>
  353. public async IAsyncEnumerable<DeepSeekStreamChunk> ChatStreamAsync(string question, string model = "deepseek-chat", float temperature = 0.7f, int maxTokens = 4000)
  354. {
  355. var messageContent = new List<object>
  356. {
  357. new { type = "text", text = question }
  358. };
  359. var request = new DeepSeekChatWithFilesRequest
  360. {
  361. Model = model,
  362. Messages = new List<FileMessage>
  363. {
  364. new FileMessage
  365. {
  366. Role = "user",
  367. Content = messageContent
  368. }
  369. },
  370. Stream = true,
  371. Temperature = temperature,
  372. MaxTokens = maxTokens,
  373. TopP = 0.9M,
  374. FrequencyPenalty = 0.2M,
  375. PresencePenalty = 0.1M,
  376. };
  377. var jsonContent = JsonSerializer.Serialize(request, new JsonSerializerOptions
  378. {
  379. PropertyNamingPolicy = JsonNamingPolicy.CamelCase
  380. });
  381. using var requestMsg = new HttpRequestMessage(HttpMethod.Post, "chat/completions")
  382. {
  383. Content = new StringContent(jsonContent, Encoding.UTF8, "application/json")
  384. };
  385. using var response = await _httpClient.SendAsync(requestMsg, HttpCompletionOption.ResponseHeadersRead).ConfigureAwait(false);
  386. response.EnsureSuccessStatusCode();
  387. using var responseStream = await response.Content.ReadAsStreamAsync().ConfigureAwait(false);
  388. using var reader = new StreamReader(responseStream, Encoding.UTF8);
  389. string? line;
  390. while ((line = await reader.ReadLineAsync().ConfigureAwait(false)) != null)
  391. {
  392. if (string.IsNullOrWhiteSpace(line))
  393. continue;
  394. if (!line.StartsWith("data: ", StringComparison.Ordinal))
  395. continue;
  396. var data = line["data: ".Length..];
  397. if (data == "[DONE]")
  398. yield break;
  399. string? reasoningPiece = null;
  400. string? contentPiece = null;
  401. try
  402. {
  403. using var jsonDoc = JsonDocument.Parse(data);
  404. if (!jsonDoc.RootElement.TryGetProperty("choices", out var choices) || choices.GetArrayLength() == 0)
  405. continue;
  406. if (!choices[0].TryGetProperty("delta", out var delta))
  407. continue;
  408. // deepseek-reasoner:思考过程在 reasoning_content,正式输出在 content
  409. if (delta.TryGetProperty("reasoning_content", out var reasoningVal))
  410. {
  411. var r = reasoningVal.GetString();
  412. if (!string.IsNullOrEmpty(r))
  413. reasoningPiece = r;
  414. }
  415. if (delta.TryGetProperty("content", out var contentVal))
  416. contentPiece = contentVal.GetString();
  417. }
  418. catch (System.Text.Json.JsonException)
  419. {
  420. }
  421. if (!string.IsNullOrEmpty(reasoningPiece))
  422. yield return new DeepSeekStreamChunk { Phase = DeepSeekStreamPhase.Reasoning, Text = reasoningPiece };
  423. if (!string.IsNullOrEmpty(contentPiece))
  424. yield return new DeepSeekStreamChunk { Phase = DeepSeekStreamPhase.Content, Text = contentPiece };
  425. }
  426. }
  427. /// <summary>
  428. /// 等待文件处理完成
  429. /// </summary>
  430. public async Task<DeepSeekFileUploadResponse> WaitForFileProcessingAsync(string fileId, int timeoutSeconds = 60)
  431. {
  432. var startTime = DateTime.UtcNow;
  433. var timeout = TimeSpan.FromSeconds(timeoutSeconds);
  434. while (DateTime.UtcNow - startTime < timeout)
  435. {
  436. var fileInfo = await GetFileInfoAsync(fileId);
  437. if (fileInfo.Status == "processed")
  438. {
  439. _logger.LogInformation("文件处理完成: {FileId}", fileId);
  440. return fileInfo;
  441. }
  442. else if (fileInfo.Status == "error")
  443. {
  444. throw new Exception($"文件处理失败: {fileId}");
  445. }
  446. // 等待2秒后重试
  447. await Task.Delay(2000);
  448. }
  449. throw new TimeoutException($"文件处理超时: {fileId}");
  450. }
  451. /// <summary>
  452. /// 根据文件名获取Content-Type
  453. /// </summary>
  454. private static string GetContentType(string fileName)
  455. {
  456. var extension = Path.GetExtension(fileName).ToLower();
  457. return extension switch
  458. {
  459. ".txt" => "text/plain",
  460. ".pdf" => "application/pdf",
  461. ".json" => "application/json",
  462. ".csv" => "text/csv",
  463. ".html" => "text/html",
  464. ".htm" => "text/html",
  465. ".md" => "text/markdown",
  466. _ => "application/octet-stream"
  467. };
  468. }
  469. #region 项目相关
  470. /// <summary>
  471. /// 读取项目内的指定文件并转换为IFormFile
  472. /// </summary>
  473. public async Task<ProjectFileReadResponse> ReadProjectFilesAsync(List<string> relativePaths)
  474. {
  475. var response = new ProjectFileReadResponse();
  476. var projectRoot = GetProjectRootPath();
  477. if (projectRoot.Contains("OASystem.Api"))
  478. {
  479. projectRoot = projectRoot.Replace("OASystem.Api", "");
  480. }
  481. _logger.LogInformation("开始读取项目文件,数量: {Count}", relativePaths.Count);
  482. foreach (var relativePath in relativePaths)
  483. {
  484. var fileInfo = new ProjectFileInfo
  485. {
  486. RelativePath = relativePath
  487. };
  488. try
  489. {
  490. var fullPath = Path.Combine(projectRoot, relativePath);
  491. fileInfo.FullPath = fullPath;
  492. if (!System.IO.File.Exists(fullPath))
  493. {
  494. throw new FileNotFoundException($"文件不存在: {relativePath}");
  495. }
  496. var fileInfoObj = new FileInfo(fullPath);
  497. fileInfo.FileSize = fileInfoObj.Length;
  498. fileInfo.LastModified = fileInfoObj.LastWriteTime;
  499. // 转换为IFormFile
  500. fileInfo.FormFile = await ConvertToFormFileAsync(fullPath);
  501. fileInfo.Success = true;
  502. _logger.LogInformation("文件读取成功: {FileName}, 大小: {Size} bytes", relativePath, fileInfo.FileSize);
  503. }
  504. catch (Exception ex)
  505. {
  506. fileInfo.Success = false;
  507. fileInfo.ErrorMessage = ex.Message;
  508. _logger.LogError(ex, "文件读取失败: {FileName}", relativePath);
  509. }
  510. response.FileInfos.Add(fileInfo);
  511. }
  512. response.Success = response.SuccessCount > 0;
  513. response.Message = $"成功读取 {response.SuccessCount} 个文件,失败 {response.FailureCount} 个";
  514. return response;
  515. }
  516. /// <summary>
  517. /// 读取项目内指定目录的文件
  518. /// </summary>
  519. public async Task<ProjectFileReadResponse> ReadProjectDirectoryAsync(string directoryPath, string searchPattern = "*.cs", bool recursive = false)
  520. {
  521. var response = new ProjectFileReadResponse();
  522. var projectRoot = GetProjectRootPath();
  523. var fullDirectoryPath = Path.Combine(projectRoot, directoryPath);
  524. _logger.LogInformation("开始读取目录文件: {Directory}, 模式: {Pattern}", directoryPath, searchPattern);
  525. if (!Directory.Exists(fullDirectoryPath))
  526. {
  527. response.Success = false;
  528. response.Message = $"目录不存在: {directoryPath}";
  529. return response;
  530. }
  531. var searchOption = recursive ? SearchOption.AllDirectories : SearchOption.TopDirectoryOnly;
  532. var files = Directory.GetFiles(fullDirectoryPath, searchPattern, searchOption);
  533. _logger.LogInformation("找到 {FileCount} 个文件", files.Length);
  534. foreach (var filePath in files)
  535. {
  536. var relativePath = Path.GetRelativePath(projectRoot, filePath);
  537. var fileInfo = new ProjectFileInfo
  538. {
  539. RelativePath = relativePath,
  540. FullPath = filePath
  541. };
  542. try
  543. {
  544. var fileInfoObj = new FileInfo(filePath);
  545. fileInfo.FileSize = fileInfoObj.Length;
  546. fileInfo.LastModified = fileInfoObj.LastWriteTime;
  547. // 转换为IFormFile
  548. fileInfo.FormFile = await ConvertToFormFileAsync(filePath);
  549. fileInfo.Success = true;
  550. _logger.LogDebug("文件读取成功: {FileName}", relativePath);
  551. }
  552. catch (Exception ex)
  553. {
  554. fileInfo.Success = false;
  555. fileInfo.ErrorMessage = ex.Message;
  556. _logger.LogError(ex, "文件读取失败: {FileName}", relativePath);
  557. }
  558. response.FileInfos.Add(fileInfo);
  559. }
  560. response.Success = response.SuccessCount > 0;
  561. response.Message = $"成功读取 {response.SuccessCount} 个文件,失败 {response.FailureCount} 个";
  562. return response;
  563. }
  564. /// <summary>
  565. /// 获取项目根目录路径
  566. /// </summary>
  567. public string GetProjectRootPath()
  568. {
  569. // 在开发环境中使用ContentRootPath,在生产环境中可能需要调整
  570. return _hostEnvironment.ContentRootPath;
  571. }
  572. /// <summary>
  573. /// 检查文件是否存在
  574. /// </summary>
  575. public bool FileExists(string relativePath)
  576. {
  577. var fullPath = Path.Combine(GetProjectRootPath(), relativePath);
  578. return System.IO.File.Exists(fullPath);
  579. }
  580. /// <summary>
  581. /// 将物理文件转换为IFormFile
  582. /// </summary>
  583. public async Task<IFormFile> ConvertToFormFileAsync(string filePath)
  584. {
  585. var fileInfo = new FileInfo(filePath);
  586. var fileName = Path.GetFileName(filePath);
  587. // 读取文件内容
  588. byte[] fileBytes;
  589. using (var fileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read))
  590. {
  591. fileBytes = new byte[fileStream.Length];
  592. await fileStream.ReadAsync(fileBytes, 0, (int)fileStream.Length);
  593. }
  594. // 创建IFormFile
  595. return new FormFile(
  596. new MemoryStream(fileBytes),
  597. 0,
  598. fileBytes.Length,
  599. "file",
  600. fileName)
  601. {
  602. Headers = new HeaderDictionary(),
  603. ContentType = GetFileContentType(fileName)
  604. };
  605. }
  606. /// <summary>
  607. /// 根据文件名获取Content-Type
  608. /// </summary>
  609. private string GetFileContentType(string fileName)
  610. {
  611. var extension = Path.GetExtension(fileName).ToLower();
  612. return extension switch
  613. {
  614. ".cs" => "text/plain",
  615. ".txt" => "text/plain",
  616. ".json" => "application/json",
  617. ".xml" => "application/xml",
  618. ".html" => "text/html",
  619. ".htm" => "text/html",
  620. ".css" => "text/css",
  621. ".js" => "application/javascript",
  622. ".ts" => "application/typescript",
  623. ".py" => "text/x-python",
  624. ".java" => "text/x-java",
  625. ".cpp" => "text/x-c++",
  626. ".c" => "text/x-c",
  627. ".h" => "text/x-c",
  628. ".md" => "text/markdown",
  629. _ => "application/octet-stream"
  630. };
  631. }
  632. /// <summary>
  633. /// 读取特定的签证申请表单文件
  634. /// </summary>
  635. /// <param name="fileName"></param>
  636. /// <returns></returns>
  637. public async Task<ProjectFileReadResponse> ReadVisaFormFileAsync(string fileName)
  638. {
  639. var specificPath = $@"OASystem\OASystem.Domain\ViewModels\VisaFormDetails\{fileName}";
  640. return await ReadProjectFilesAsync(new List<string> { specificPath });
  641. }
  642. /// <summary>
  643. /// 读取OASystem项目中的所有CS文件
  644. /// </summary>
  645. public async Task<ProjectFileReadResponse> ReadOASystemFilesAsync(string searchPattern = "*.cs", bool recursive = true)
  646. {
  647. return await ReadProjectDirectoryAsync(@"OASystem", searchPattern, recursive);
  648. }
  649. /// <summary>
  650. /// 读取指定域模型中的CS文件
  651. /// </summary>
  652. public async Task<ProjectFileReadResponse> ReadDomainViewModelsAsync(string searchPattern = "*.cs", bool recursive = true)
  653. {
  654. return await ReadProjectDirectoryAsync(@"OASystem\OASystem.Domain\ViewModels", searchPattern, recursive);
  655. }
  656. /// <summary>
  657. /// 读取签证表单相关的所有CS文件
  658. /// </summary>
  659. public async Task<ProjectFileReadResponse> ReadVisaFormFilesAsync(string searchPattern = "*.cs", bool recursive = true)
  660. {
  661. return await ReadProjectDirectoryAsync(@"OASystem\OASystem.Domain\ViewModels\VisaFormDetails", searchPattern, recursive);
  662. }
  663. #endregion
  664. }
  665. #region 私有实体类
  666. /// <summary>
  667. /// DeepSeek 聊天响应(用于反序列化)
  668. /// </summary>
  669. internal class DeepSeekResponse
  670. {
  671. [JsonPropertyName("choices")]
  672. public List<Choice> Choices { get; set; }
  673. [JsonPropertyName("usage")]
  674. public Usage Usage { get; set; }
  675. }
  676. internal class Choice
  677. {
  678. [JsonPropertyName("message")]
  679. public Message Message { get; set; }
  680. }
  681. internal class Message
  682. {
  683. [JsonPropertyName("content")]
  684. public string Content { get; set; }
  685. }
  686. internal class Usage
  687. {
  688. [JsonPropertyName("total_tokens")]
  689. public int TotalTokens { get; set; }
  690. }
  691. #endregion
  692. }