KiMiApi.cs 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  1. using OASystem.Domain.Dtos.Groups;
  2. using OASystem.Domain.Dtos.KiMi;
  3. using System.Net.Http.Headers;
  4. namespace OASystem.API.OAMethodLib.KiMiApi
  5. {
  6. public class KiMiApiClient
  7. {
  8. //读取配置文件
  9. private KiMiSetting _kimiSetting { get; set; }
  10. private readonly HttpClient _httpClient;
  11. public KiMiApiClient()
  12. {
  13. _httpClient = new HttpClient();
  14. _httpClient.Timeout = TimeSpan.FromMinutes(30);
  15. _kimiSetting = AutofacIocManager.Instance.GetService<KiMiSetting>();
  16. // 设置公共请求头
  17. _httpClient.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", _kimiSetting.Key);
  18. }
  19. //private string apiKey = "sk-AY1Sv4rLnSUgGGHcC8SGSWYYKzGID7leZJcFfxAYozLC8dIc";
  20. //private string baseUrl = "https://api.moonshot.cn/v1";
  21. public async Task<string> UploadFileAsync(IFormFile file)
  22. {
  23. string message = string.Empty;
  24. byte[] fileBytes = null;
  25. using (var memoryStream = new MemoryStream())
  26. {
  27. await file.CopyToAsync(memoryStream);
  28. fileBytes = memoryStream.ToArray();
  29. }
  30. // 上传文件
  31. var fileContent = new ByteArrayContent(fileBytes);
  32. var fileFormData = new MultipartFormDataContent
  33. {
  34. { fileContent, "file", file.FileName },
  35. { new StringContent("file-extract"), "purpose" }
  36. };
  37. HttpResponseMessage fileResponse = await _httpClient.PostAsync($"{_kimiSetting.BaseUrl}/files", fileFormData);
  38. string fileResponseContent = await fileResponse.Content.ReadAsStringAsync();
  39. if (!fileResponse.IsSuccessStatusCode)
  40. {
  41. throw new Exception($"Failed to upload file: {fileResponseContent}");
  42. }
  43. dynamic fileObject = JsonConvert.DeserializeObject(fileResponseContent);
  44. string fileId = fileObject.id;
  45. HttpResponseMessage contentResponse = await _httpClient.GetAsync($"{_kimiSetting.BaseUrl}/files/{fileId}/content");
  46. string fileContentText = await contentResponse.Content.ReadAsStringAsync();
  47. if (!contentResponse.IsSuccessStatusCode)
  48. {
  49. throw new Exception($"Failed to get file content: {fileContentText}");
  50. }
  51. message = fileContentText;
  52. return message;
  53. }
  54. public async Task<List<SeedMessages>> UploadFilesAsync(List<IFormFile> files)
  55. {
  56. List<SeedMessages> result = new List<SeedMessages>();
  57. foreach (IFormFile file in files)
  58. {
  59. var fileConter = await this.UploadFileAsync(file);
  60. result.Add(new SeedMessages() { Role = KimiRole.system , Content = fileConter });
  61. }
  62. return result;
  63. }
  64. public async Task<string> SeedMessage(List<SeedMessages> messages)
  65. {
  66. string completionResponseContent = await SeedAsync(messages);
  67. // 解析返回的完成结果
  68. var completion = JObject.Parse(completionResponseContent);
  69. string reply = completion["choices"][0]["message"].ToString();
  70. return reply;
  71. }
  72. public async Task<KiMiRoot> SeedMessageByFullConterObject(List<SeedMessages> messages)
  73. {
  74. var respStr = await SeedAsync(messages);
  75. return JsonConvert.DeserializeObject<KiMiRoot>(respStr);
  76. }
  77. private async Task<string> SeedAsync(List<SeedMessages> messages)
  78. {
  79. var completionRequest = new
  80. {
  81. model = _kimiSetting.Model,
  82. messages = messages.Select(x => new
  83. {
  84. Role = StringEnumHelper.ToStringValue(x.Role),
  85. x.Content
  86. }),
  87. temperature = 1,
  88. max_tokens = 4096,
  89. // thinking = new { type = "disabled" } // 禁用 thinking 模式
  90. };
  91. var completionJson = JsonConvert.SerializeObject(completionRequest);
  92. var completionContent = new StringContent(completionJson, Encoding.UTF8, "application/json");
  93. HttpResponseMessage completionResponse = await _httpClient.PostAsync($"{_kimiSetting.BaseUrl}/chat/completions", completionContent);
  94. string completionResponseContent = await completionResponse.Content.ReadAsStringAsync();
  95. if (!completionResponse.IsSuccessStatusCode)
  96. {
  97. throw new Exception($"Failed to seed message: {completionResponseContent}");
  98. }
  99. return completionResponseContent;
  100. }
  101. }
  102. public class SeedMessages
  103. {
  104. public KimiRole Role { get; set; }
  105. public string Content { get; set; }
  106. }
  107. public class KiMiRoot
  108. {
  109. public string Id { get; set; }
  110. public string Object { get; set; }
  111. public long Created { get; set; }
  112. public string Model { get; set; }
  113. public List<Choice> Choices { get; set; }
  114. public Usage Usage { get; set; }
  115. }
  116. public class Choice
  117. {
  118. [JsonProperty("index")]
  119. public int Index { get; set; }
  120. public ReturnMessage Message { get; set; }
  121. [JsonProperty("finish_reason")]
  122. public string FinishReason { get; set; }
  123. }
  124. public class Usage
  125. {
  126. public int PromptTokens { get; set; }
  127. public int CompletionTokens { get; set; }
  128. public int TotalTokens { get; set; }
  129. }
  130. public class ReturnMessage
  131. {
  132. public string Role { get; set; }
  133. public string Content { get; set; }
  134. }
  135. public enum KimiRole
  136. {
  137. system,
  138. user
  139. }
  140. public static class StringEnumHelper
  141. {
  142. public static string ToStringValue(KimiRole value)
  143. {
  144. return value switch
  145. {
  146. KimiRole.system => "system",
  147. KimiRole.user => "user",
  148. _ => throw new ArgumentOutOfRangeException(nameof(value), value, null)
  149. };
  150. }
  151. }
  152. }