SysLangTextService.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432
  1. // Admin.NET 项目的版权、商标、专利和其他相关权利均受相应法律法规的保护。使用本项目应遵守相关法律法规和许可证的要求。
  2. //
  3. // 本项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。许可证位于源代码树根目录中的 LICENSE-MIT 和 LICENSE-APACHE 文件。
  4. //
  5. // 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动!任何基于本项目二次开发而产生的一切法律纠纷和责任,我们不承担任何责任!
  6. using AngleSharp.Dom;
  7. using Microsoft.AspNetCore.Components.Forms;
  8. using Newtonsoft.Json;
  9. namespace Admin.NET.Core.Service;
  10. /// <summary>
  11. /// 翻译服务 🧩
  12. /// </summary>
  13. [ApiDescriptionSettings(Order = 100, Description = "翻译服务")]
  14. public partial class SysLangTextService : IDynamicApiController, ITransient
  15. {
  16. private readonly SqlSugarRepository<SysLangText> _sysLangTextRep;
  17. private readonly ISqlSugarClient _sqlSugarClient;
  18. private readonly SysLangTextCacheService _sysLangTextCacheService;
  19. public SysLangTextService(
  20. SqlSugarRepository<SysLangText> sysLangTextRep,
  21. SysLangTextCacheService sysLangTextCacheService,
  22. ISqlSugarClient sqlSugarClient)
  23. {
  24. _sysLangTextRep = sysLangTextRep;
  25. _sqlSugarClient = sqlSugarClient;
  26. _sysLangTextCacheService = sysLangTextCacheService;
  27. }
  28. /// <summary>
  29. /// 分页查询翻译表 🔖
  30. /// </summary>
  31. /// <param name="input"></param>
  32. /// <returns></returns>
  33. [DisplayName("分页查询翻译表")]
  34. [ApiDescriptionSettings(Name = "Page"), HttpPost]
  35. public async Task<SqlSugarPagedList<SysLangTextOutput>> Page(PageSysLangTextInput input)
  36. {
  37. input.Keyword = input.Keyword?.Trim();
  38. var query = _sysLangTextRep.AsQueryable()
  39. .WhereIF(!string.IsNullOrWhiteSpace(input.Keyword), u => u.EntityName.Contains(input.Keyword) || u.FieldName.Contains(input.Keyword) || u.LangCode.Contains(input.Keyword) || u.Content.Contains(input.Keyword))
  40. .WhereIF(!string.IsNullOrWhiteSpace(input.EntityName), u => u.EntityName.Contains(input.EntityName.Trim()))
  41. .WhereIF(!string.IsNullOrWhiteSpace(input.FieldName), u => u.FieldName.Contains(input.FieldName.Trim()))
  42. .WhereIF(!string.IsNullOrWhiteSpace(input.LangCode), u => u.LangCode.Contains(input.LangCode.Trim()))
  43. .WhereIF(!string.IsNullOrWhiteSpace(input.Content), u => u.Content.Contains(input.Content.Trim()))
  44. .WhereIF(input.EntityId != null, u => u.EntityId == input.EntityId)
  45. .Select<SysLangTextOutput>();
  46. return await query.OrderBuilder(input).ToPagedListAsync(input.Page, input.PageSize);
  47. }
  48. [DisplayName("获取翻译表")]
  49. [ApiDescriptionSettings(Name = "List"), HttpPost]
  50. public async Task<List<SysLangTextOutput>> List(ListSysLangTextInput input)
  51. {
  52. var query = _sysLangTextRep.AsQueryable()
  53. .Where(u => u.EntityName == input.EntityName.Trim() && u.FieldName == input.FieldName.Trim() && u.EntityId == input.EntityId)
  54. .WhereIF(!string.IsNullOrWhiteSpace(input.LangCode), u => u.LangCode == input.LangCode.Trim())
  55. .Select<SysLangTextOutput>();
  56. return await query.ToListAsync();
  57. }
  58. /// <summary>
  59. /// 获取翻译表详情 ℹ️
  60. /// </summary>
  61. /// <param name="input"></param>
  62. /// <returns></returns>
  63. [DisplayName("获取翻译表详情")]
  64. [ApiDescriptionSettings(Name = "Detail"), HttpGet]
  65. public async Task<SysLangText> Detail([FromQuery] QueryByIdSysLangTextInput input)
  66. {
  67. return await _sysLangTextRep.GetFirstAsync(u => u.Id == input.Id);
  68. }
  69. /// <summary>
  70. /// 增加翻译表 ➕
  71. /// </summary>
  72. /// <param name="input"></param>
  73. /// <returns></returns>
  74. [DisplayName("增加翻译表")]
  75. [ApiDescriptionSettings(Name = "Add"), HttpPost]
  76. public async Task<long> Add(AddSysLangTextInput input)
  77. {
  78. var entity = input.Adapt<SysLangText>();
  79. return await _sysLangTextRep.InsertAsync(entity) ? entity.Id : 0;
  80. }
  81. /// <summary>
  82. /// 更新翻译表 ✏️
  83. /// </summary>
  84. /// <param name="input"></param>
  85. /// <returns></returns>
  86. [DisplayName("更新翻译表")]
  87. [ApiDescriptionSettings(Name = "Update"), HttpPost]
  88. public async Task Update(UpdateSysLangTextInput input)
  89. {
  90. var entity = input.Adapt<SysLangText>();
  91. await _sysLangTextRep.AsUpdateable(entity)
  92. .ExecuteCommandAsync();
  93. _sysLangTextCacheService.UpdateCache(entity.EntityName, entity.FieldName, entity.EntityId, entity.LangCode, entity.Content);
  94. }
  95. /// <summary>
  96. /// 删除翻译表 ❌
  97. /// </summary>
  98. /// <param name="input"></param>
  99. /// <returns></returns>
  100. [DisplayName("删除翻译表")]
  101. [ApiDescriptionSettings(Name = "Delete"), HttpPost]
  102. public async Task Delete(DeleteSysLangTextInput input)
  103. {
  104. var entity = await _sysLangTextRep.GetFirstAsync(u => u.Id == input.Id) ?? throw Oops.Oh(ErrorCodeEnum.D1002);
  105. await _sysLangTextRep.DeleteAsync(entity); //真删除
  106. _sysLangTextCacheService.DeleteCache(entity.EntityName, entity.FieldName, entity.EntityId, entity.LangCode);
  107. }
  108. /// <summary>
  109. /// 批量删除翻译表 ❌
  110. /// </summary>
  111. /// <param name="input"></param>
  112. /// <returns></returns>
  113. [DisplayName("批量删除翻译表")]
  114. [ApiDescriptionSettings(Name = "BatchDelete"), HttpPost]
  115. public async Task BatchDelete([Required(ErrorMessage = "主键列表不能为空")] List<DeleteSysLangTextInput> input)
  116. {
  117. var exp = Expressionable.Create<SysLangText>();
  118. foreach (var row in input) exp = exp.Or(it => it.Id == row.Id);
  119. var list = await _sysLangTextRep.AsQueryable().Where(exp.ToExpression()).ToListAsync();
  120. await _sysLangTextRep.DeleteAsync(list); //真删除
  121. foreach (var item in list)
  122. {
  123. _sysLangTextCacheService.DeleteCache(item.EntityName, item.FieldName, item.EntityId, item.LangCode);
  124. }
  125. }
  126. private static readonly object _sysLangTextBatchSaveLock = new object();
  127. /// <summary>
  128. /// 批量保存翻译表 ✏️
  129. /// </summary>
  130. /// <param name="input"></param>
  131. /// <returns></returns>
  132. [DisplayName("批量保存翻译表")]
  133. [ApiDescriptionSettings(Name = "BatchSave"), HttpPost]
  134. public void BatchSave([Required(ErrorMessage = "列表不能为空")] List<ImportSysLangTextInput> input)
  135. {
  136. lock (_sysLangTextBatchSaveLock)
  137. {
  138. // 校验并过滤必填基本类型为null的字段
  139. var rows = input.Where(x =>
  140. {
  141. if (!string.IsNullOrWhiteSpace(x.Error)) return false;
  142. if (x.EntityId == null)
  143. {
  144. x.Error = "所属实体ID不能为空";
  145. return false;
  146. }
  147. return true;
  148. }).Adapt<List<SysLangText>>();
  149. var storageable = _sysLangTextRep.Context.Storageable(rows)
  150. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.EntityName), "所属实体名不能为空")
  151. .SplitError(it => it.Item.EntityName?.Length > 255, "所属实体名长度不能超过255个字符")
  152. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.FieldName), "字段名不能为空")
  153. .SplitError(it => it.Item.FieldName?.Length > 255, "字段名长度不能超过255个字符")
  154. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.LangCode), "语言代码不能为空")
  155. .SplitError(it => it.Item.LangCode?.Length > 255, "语言代码长度不能超过255个字符")
  156. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.Content), "翻译内容不能为空")
  157. .WhereColumns(it => new { it.EntityId, it.EntityName, it.FieldName, it.LangCode })
  158. .SplitInsert(it => it.NotAny())
  159. .SplitUpdate(it => it.Any())
  160. .ToStorage();
  161. storageable.AsInsertable.ExecuteCommand();// 不存在插入
  162. storageable.AsUpdateable.UpdateColumns(it => new
  163. {
  164. it.EntityName,
  165. it.EntityId,
  166. it.FieldName,
  167. it.LangCode,
  168. it.Content,
  169. }).ExecuteCommand();// 存在更新
  170. foreach (var item in rows)
  171. {
  172. _sysLangTextCacheService.DeleteCache(item.EntityName, item.FieldName, item.EntityId, item.LangCode);
  173. }
  174. if (storageable.ErrorList.Any())
  175. {
  176. throw Oops.Oh($"处理过程中出现以下错误:{string.Join(";", storageable.ErrorList.Distinct())}");
  177. }
  178. }
  179. }
  180. /// <summary>
  181. /// 导出翻译表记录 🔖
  182. /// </summary>
  183. /// <param name="input"></param>
  184. /// <returns></returns>
  185. [DisplayName("导出翻译表记录")]
  186. [ApiDescriptionSettings(Name = "Export"), HttpPost, NonUnify]
  187. public async Task<IActionResult> Export(PageSysLangTextInput input)
  188. {
  189. var list = (await Page(input)).Items?.Adapt<List<ExportSysLangTextOutput>>() ?? new();
  190. if (input.SelectKeyList?.Count > 0) list = list.Where(x => input.SelectKeyList.Contains(x.Id)).ToList();
  191. return ExcelHelper.ExportTemplate(list, "翻译表导出记录");
  192. }
  193. /// <summary>
  194. /// 下载翻译表数据导入模板 ⬇️
  195. /// </summary>
  196. /// <returns></returns>
  197. [DisplayName("下载翻译表数据导入模板")]
  198. [ApiDescriptionSettings(Name = "Import"), HttpGet, NonUnify]
  199. public IActionResult DownloadTemplate()
  200. {
  201. return ExcelHelper.ExportTemplate(new List<ExportSysLangTextOutput>(), "翻译表导入模板");
  202. }
  203. private static readonly object _sysLangTextImportLock = new object();
  204. /// <summary>
  205. /// 导入翻译表记录 💾
  206. /// </summary>
  207. /// <returns></returns>
  208. [DisplayName("导入翻译表记录")]
  209. [ApiDescriptionSettings(Name = "Import"), HttpPost, NonUnify, UnitOfWork]
  210. public IActionResult ImportData([Required] IFormFile file)
  211. {
  212. lock (_sysLangTextImportLock)
  213. {
  214. var stream = ExcelHelper.ImportData<ImportSysLangTextInput, SysLangText>(file, (list, markerErrorAction) =>
  215. {
  216. _sqlSugarClient.Utilities.PageEach(list, 2048, pageItems =>
  217. {
  218. // 校验并过滤必填基本类型为null的字段
  219. var rows = pageItems.Where(x =>
  220. {
  221. if (!string.IsNullOrWhiteSpace(x.Error)) return false;
  222. if (x.EntityId == null)
  223. {
  224. x.Error = "所属实体ID不能为空";
  225. return false;
  226. }
  227. return true;
  228. }).Adapt<List<SysLangText>>();
  229. var storageable = _sysLangTextRep.Context.Storageable(rows)
  230. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.EntityName), "所属实体名不能为空")
  231. .SplitError(it => it.Item.EntityName?.Length > 255, "所属实体名长度不能超过255个字符")
  232. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.FieldName), "字段名不能为空")
  233. .SplitError(it => it.Item.FieldName?.Length > 255, "字段名长度不能超过255个字符")
  234. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.LangCode), "语言代码不能为空")
  235. .SplitError(it => it.Item.LangCode?.Length > 255, "语言代码长度不能超过255个字符")
  236. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.Content), "翻译内容不能为空")
  237. .SplitError(it => it.Item.Content?.Length > 255, "翻译内容长度不能超过255个字符")
  238. .WhereColumns(it => new { it.EntityId, it.EntityName, it.FieldName, it.LangCode })
  239. .SplitInsert(it => it.NotAny())
  240. .SplitUpdate(it => it.Any())
  241. .ToStorage();
  242. storageable.AsInsertable.ExecuteCommand();// 不存在插入
  243. storageable.AsUpdateable.UpdateColumns(it => new
  244. {
  245. it.EntityName,
  246. it.EntityId,
  247. it.FieldName,
  248. it.LangCode,
  249. it.Content,
  250. }).ExecuteCommand();// 存在更新
  251. foreach (var item in rows)
  252. {
  253. _sysLangTextCacheService.DeleteCache(item.EntityName, item.FieldName, item.EntityId, item.LangCode);
  254. }
  255. // 标记错误信息
  256. markerErrorAction.Invoke(storageable, pageItems, rows);
  257. });
  258. });
  259. return stream;
  260. }
  261. }
  262. /// <summary>
  263. /// DEEPSEEK 翻译接口
  264. /// </summary>
  265. /// <returns></returns>
  266. [DisplayName("DEEPSEEK 翻译接口")]
  267. [ApiDescriptionSettings(Name = "AiTranslateText"), HttpPost]
  268. public async Task<string> AiTranslateText(AiTranslateTextInput input)
  269. {
  270. // 需要先把DeepSeek.example复制改名为DeepSeek.json文件,添加你的 API KEY
  271. var deepSeekOptions = App.GetConfig<DeepSeekOptions>("DeepSeekSettings", true);
  272. if (deepSeekOptions == null)
  273. {
  274. throw new InvalidOperationException("DeepSeek.json文件 未定义");
  275. }
  276. if (string.IsNullOrEmpty(deepSeekOptions.ApiKey))
  277. {
  278. throw new InvalidOperationException("环境变量 DEEPSEEK_API_KEY 未定义");
  279. }
  280. using (HttpClient client = new HttpClient())
  281. {
  282. // 构建请求头
  283. client.DefaultRequestHeaders.Add("Authorization", $"Bearer {deepSeekOptions.ApiKey}");
  284. // 构建系统提示词
  285. string systemPrompt = BuildSystemPrompt(deepSeekOptions.SourceLang, input.TargetLang);
  286. // 构建请求体
  287. var requestBody = new
  288. {
  289. model = "deepseek-chat",
  290. messages = new[]
  291. {
  292. new { role = "system", content = systemPrompt },
  293. new { role = "user", content = input.OriginalText }
  294. },
  295. temperature = 0.3,
  296. max_tokens = 2000
  297. };
  298. // 使用 Newtonsoft.Json 序列化
  299. var json = JsonConvert.SerializeObject(requestBody);
  300. var content = new StringContent(json, Encoding.UTF8, "application/json");
  301. // 发送请求
  302. HttpResponseMessage response = await client.PostAsync(deepSeekOptions.ApiUrl, content);
  303. // 处理响应
  304. string responseBody = await response.Content.ReadAsStringAsync();
  305. if (!response.IsSuccessStatusCode)
  306. {
  307. // 使用 Newtonsoft.Json 反序列化错误响应
  308. var errorResponse = JsonConvert.DeserializeObject<ErrorResponse>(responseBody);
  309. string errorMsg = errorResponse?.error?.message ?? $"HTTP {response.StatusCode}: {response.ReasonPhrase}";
  310. throw new HttpRequestException($"翻译API返回错误:{errorMsg}");
  311. }
  312. // 解析有效响应
  313. var result = JsonConvert.DeserializeObject<TranslationResponse>(responseBody);
  314. if (result?.choices == null || result.choices.Length == 0 ||
  315. result.choices[0]?.message?.content == null)
  316. {
  317. throw new InvalidOperationException("API返回无效的翻译结果");
  318. }
  319. return result.choices[0].message.content.Trim();
  320. }
  321. }
  322. // JSON 响应模型
  323. private class TranslationResponse
  324. {
  325. public Choice[] choices { get; set; }
  326. }
  327. private class Choice
  328. {
  329. public Message message { get; set; }
  330. }
  331. private class Message
  332. {
  333. public string content { get; set; }
  334. }
  335. private class ErrorResponse
  336. {
  337. public ErrorInfo error { get; set; }
  338. }
  339. private class ErrorInfo
  340. {
  341. public string message { get; set; }
  342. }
  343. /// <summary>
  344. /// 生成提示词
  345. /// </summary>
  346. /// <param name="targetLang"></param>
  347. /// <returns></returns>
  348. private static string BuildSystemPrompt(string sourceLang, string targetLang)
  349. {
  350. return $@"作为企业软件系统专业翻译,严格遵守以下铁律:
  351. ■ 核心原则
  352. 1. 严格逐符号翻译({sourceLang}→{targetLang})
  353. 2. 禁止添加/删除/改写任何内容
  354. 3. 保持批量翻译的编号格式
  355. ■ 符号保留规则
  356. ! 所有符号必须原样保留:
  357. • 编程符号:\${{ }} <% %> @ # & |
  358. • UI占位符:{{0}} %s [ ]
  359. • 货币单位:¥100.00 kg cm²
  360. • 中文符号:【 】 《 》 :
  361. ■ 中文符号位置规范
  362. # 三级处理机制:
  363. 1. 成对符号必须保持完整结构:
  364. ✓ 正确:【Warning】Text
  365. ✗ 禁止:Warning【 】Text
  366. 2. 独立符号位置:
  367. • 优先句尾 → Text】?
  368. • 次选句首 → 】Text?
  369. • 禁止句中 → Text】Text?
  370. 3. 跨字符串符号处理:
  371. • 前段含【时 → 保留在段尾(""Synchronize【"")
  372. • 后段含】时 → 保留在段首(""】authorization data?"")
  373. • 符号后接字母时添加空格:】 Authorization
  374. ■ 语法规范
  375. • 外文 → 被动语态(""Item was created"")
  376. • 中文 → 主动语态(""已创建项目"")
  377. • 禁止推测上下文(只翻译当前字符串内容)
  378. ■ 错误预防(绝对禁止)
  379. ✗ 将中文符号改为西式符号(】→])
  380. ✗ 移动非中文符号位置
  381. ✗ 添加原文不存在的内容
  382. ✗ 合并/拆分原始字符串
  383. ■ 批量处理
  384. ▸ 严格保持原始JSON结构
  385. ▸ 语言键名精确匹配(zh-cn/en/it等)";
  386. }
  387. }