KiMiApi.cs 5.8 KB

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