FileImportController.cs 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455
  1. using Microsoft.AspNetCore.Http;
  2. using Microsoft.AspNetCore.Mvc;
  3. using Microsoft.Extensions.Options;
  4. using Ropin.Core.Extensions.Redis;
  5. using Ropin.Core.Extensions;
  6. using Ropin.Inspection.Api.Common.Options;
  7. using Ropin.Inspection.Api.Common.Token;
  8. using Ropin.Inspection.Service.Interface;
  9. using Ropin.Inspection.Service.SYS.Interface;
  10. using Ropin.Inspection.Service;
  11. using System.Net.Http;
  12. using Ropin.Inspection.Common.Helper;
  13. using NPOI.HSSF.UserModel;
  14. using NPOI.SS.UserModel;
  15. using NPOI.XSSF.UserModel;
  16. using Ropin.Inspection.Api.Common;
  17. using System.ComponentModel.DataAnnotations;
  18. using System.Linq;
  19. using System.Threading.Tasks;
  20. using log4net;
  21. using System.Collections.Generic;
  22. using System.ComponentModel;
  23. using System;
  24. using Ropin.Inspection.Model.ViewModel.ISP;
  25. using NPOI.Util;
  26. using System.Reflection;
  27. using Ropin.Inspection.Model.ViewModel;
  28. using Ropin.Inspection.Model;
  29. using Ropin.Inspection.Model.SearchModel;
  30. using Ropin.Inspection.Model.Entities;
  31. using Ropin.Inspection.Common.Accessor.Interface;
  32. using System.Security.Claims;
  33. using NPOI.OpenXmlFormats.Dml.Diagram;
  34. using NPOI.SS.Formula.Functions;
  35. using NPOI.SS.Util;
  36. using Newtonsoft.Json;
  37. using System.Xml.Linq;
  38. namespace Ropin.Inspection.Api.Controllers.Base
  39. {
  40. public class FileImportController : BaseController
  41. {
  42. private static readonly ILog log = LogManager.GetLogger(typeof(FileImportController));
  43. private readonly IClaimsAccessor _claims;
  44. private readonly ITsysOrganizeService _tsysOrganizeService;
  45. private readonly ITpntStoreService _TpntStoreService;
  46. private readonly ITispContentService _tispContentService;
  47. private readonly ITispContentGroupService _tispContentGroupService;
  48. private readonly ITispContentGroupItemService _tispContentGroupItemService;
  49. private readonly ITdevDevStoreService _tdevDevStoreService;
  50. public FileImportController(IClaimsAccessor claims,ITsysOrganizeService tsysOrganizeService, ITpntStoreService TpntStoreService, ITispContentService tispContentService, ITispContentGroupService tispContentGroupService, ITispContentGroupItemService tispContentGroupItemService, ITdevDevStoreService tdevDevStoreService)
  51. {
  52. _claims = claims;
  53. _tsysOrganizeService = tsysOrganizeService;
  54. _TpntStoreService = TpntStoreService;
  55. _tispContentService = tispContentService;
  56. _tispContentGroupService = tispContentGroupService;
  57. _tispContentGroupItemService = tispContentGroupItemService;
  58. _tdevDevStoreService = tdevDevStoreService;
  59. }
  60. /// <summary>
  61. /// 导入巡检内容前的判断
  62. /// </summary>
  63. /// <param name="orgCode">组织ID</param>
  64. /// <param name="storeCode">业主ID</param>
  65. /// <param name="file"></param>
  66. /// <returns></returns>
  67. [HttpPost("JudgmentImportIspContent")]
  68. public async Task<ApiResult> JudgmentImportIspContent(string orgCode, string storeCode, [Required] IFormFile file)
  69. {
  70. if (string.IsNullOrEmpty(orgCode) || string.IsNullOrEmpty(storeCode))
  71. {
  72. return new ApiResult(ReturnCode.ArgsError);
  73. }
  74. if (file == null)
  75. {
  76. return new ApiResult(ReturnCode.ArgsError, "请上传导入文件");
  77. }
  78. try
  79. {
  80. string msg = "";
  81. Guid OrgId = Guid.Parse(orgCode);
  82. Guid StoreId = Guid.Parse(storeCode);
  83. TsysOrganizeViewModel orgEntity = await _tsysOrganizeService.GetByIdAsync(OrgId);
  84. TpntStoreViewModel storeEntity = await _TpntStoreService.GetByIdAsync(StoreId);
  85. if (orgEntity == null || storeEntity == null)
  86. {
  87. msg = "业主和组织获取失败";
  88. return new ApiResult(ReturnCode.GeneralError, msg);
  89. }
  90. List<ISPFIleImport> researchExcels = new List<ISPFIleImport>();
  91. IWorkbook wk = null;
  92. var exName = file.FileName.Split('.').Last().ToLower();
  93. if (exName == "xlsx")
  94. {
  95. wk = new XSSFWorkbook(file.OpenReadStream());
  96. }
  97. else
  98. {
  99. wk = new HSSFWorkbook(file.OpenReadStream());
  100. }
  101. //获取第一个sheet
  102. ISheet sheet = wk.GetSheetAt(0);
  103. //获取第一行-设备
  104. IRow headrow0 = sheet.GetRow(0);
  105. //获取第二行-业主
  106. IRow headrow2 = sheet.GetRow(1);
  107. //获取第三行-组织
  108. IRow headrow3 = sheet.GetRow(2);
  109. IRow headrow4= sheet.GetRow(3);
  110. IRow headrow5 = sheet.GetRow(4);
  111. if (headrow0 == null || headrow2 == null || headrow3 == null || headrow4 == null || headrow5 == null)
  112. {
  113. return new ApiResult(ReturnCode.GeneralError, "请确认表格内是否有数据数据");
  114. }
  115. string devCell0 = headrow0.Cells[0].ToString();
  116. var nameCell2 = headrow2.Cells.Where(x => x.ToString() == "使用部门:").FirstOrDefault();
  117. var valCell2 = headrow2.Cells[2].ToString();
  118. var nameCell3 = headrow3.Cells.Where(x => x.ToString() == "服务公司:").FirstOrDefault();
  119. var valCell3 = headrow3.Cells[2].ToString();
  120. if (nameCell2 is null || nameCell3 is null)
  121. {
  122. msg = "导入格式有误,请使用模板进行导入";
  123. return new ApiResult(ReturnCode.ArgsError, msg);
  124. }
  125. if ((storeEntity.C_Name.Trim() != valCell2 && storeEntity.C_SName?.Trim() != valCell2) || (orgEntity.C_SName?.Trim() != valCell3 && orgEntity.C_Name.Trim() != valCell3))
  126. {
  127. msg = "业主和组织验证失败";
  128. return new ApiResult(ReturnCode.ArgsError, msg);
  129. }
  130. if (!string.IsNullOrEmpty(devCell0))
  131. {
  132. var devArray = devCell0.Split("点检记录表").ToList();
  133. try
  134. {
  135. var arr0 = devArray[0].ToString();
  136. if (string.IsNullOrEmpty(arr0))
  137. {
  138. msg = "获取设备名称错误,格式:【XXX点检记录表(设备编号:XXX)】";
  139. return new ApiResult(ReturnCode.ArgsError, msg);
  140. }
  141. TdevDevStoreSearchModel searchDev = new TdevDevStoreSearchModel();
  142. searchDev.C_StoreCode = storeCode;
  143. searchDev.C_Name = arr0;
  144. var DevData = await _tdevDevStoreService.GetConditionAsync(searchDev);
  145. if (DevData == null || DevData.Count() == 0 || (DevData.Count() == 1 && DevData.First() == null) || DevData.Count() > 1)
  146. {
  147. msg = $"请确认设备【{arr0}】名称是否正确或者是否存在于当前业主【{valCell2}】下";
  148. return new ApiResult(ReturnCode.ArgsError, msg);
  149. }
  150. var devModel = DevData.FirstOrDefault();
  151. msg = $"请确认是否导入数据到 组织:{valCell3},业主:{valCell2},设备:{devModel.C_Name}【{devModel.C_NumberCode}】 下吗?";
  152. var data = new
  153. {
  154. devCode = devModel.C_ID,
  155. Message=msg,
  156. };
  157. return new ApiResult<object>(data, ReturnCode.Success);
  158. }
  159. catch (Exception)
  160. {
  161. msg = "获取首行设备错误";
  162. return new ApiResult(ReturnCode.GeneralError, msg);
  163. }
  164. }
  165. else
  166. {
  167. msg = "请填写表格表头,格式:【XXX点检记录表(设备编号:XXX)】";
  168. return new ApiResult(ReturnCode.ArgsError, msg);
  169. }
  170. }
  171. catch (Exception ex)
  172. {
  173. return new ApiResult(ReturnCode.GeneralError, "导入模版格式不对!");
  174. }
  175. }
  176. /// <summary>
  177. /// 导入巡检内容
  178. /// </summary>
  179. /// <param name="orgCode"></param>
  180. /// <param name="storeCode"></param>
  181. /// <param name="devCode">设备ID</param>
  182. /// <param name="file"></param>
  183. /// <returns></returns>
  184. [HttpPost("ImportIspContent")]
  185. public async Task<ApiResult> ImportIspContent(string orgCode, string storeCode, string devCode, [Required] IFormFile file)
  186. {
  187. if (string.IsNullOrEmpty(orgCode) || string.IsNullOrEmpty(storeCode))
  188. {
  189. return new ApiResult(ReturnCode.ArgsError);
  190. }
  191. if (file == null)
  192. {
  193. return new ApiResult(ReturnCode.ArgsError,"请上传导入文件");
  194. }
  195. try
  196. {
  197. List<ISPFIleImport> researchExcels = new List<ISPFIleImport>();
  198. IWorkbook wk = null;
  199. var exName = file.FileName.Split('.').Last().ToLower();
  200. if (exName == "xlsx")
  201. {
  202. wk = new XSSFWorkbook(file.OpenReadStream());
  203. }
  204. else
  205. {
  206. wk = new HSSFWorkbook(file.OpenReadStream());
  207. }
  208. //获取第一个sheet
  209. ISheet sheet = wk.GetSheetAt(0);
  210. // 存储已处理的合并区域
  211. List<CellRangeAddress> processedMergedRegions = new List<CellRangeAddress>();
  212. //获取第四行-标题行
  213. IRow headrow4 = sheet.GetRow(3);
  214. var researchExcel = new ISPFIleImport();
  215. var units = new List<string>();
  216. //读取每行,从第五行起
  217. for (int r = 4; r <= sheet.LastRowNum; r++)
  218. {
  219. var copyExcel = researchExcel.Copy();
  220. //获取当前行
  221. IRow row = sheet.GetRow(r);
  222. // 判断当前行是否有内容
  223. bool hasContent = row != null && row.Cells.Any(cell =>
  224. {
  225. if (cell == null) return false;
  226. var cellValue = cell.ToString().Trim();
  227. return !string.IsNullOrEmpty(cellValue);
  228. });
  229. if (!hasContent)
  230. {
  231. // 如果当前行没有内容,跳过当前循环
  232. continue;
  233. }
  234. //读取每列
  235. for (int j = 0; j < row.Cells.Count; j++)
  236. {
  237. ICell cell = row.GetCell(j); //一个单元格
  238. var cellValue = cell.ToString();
  239. var properties = copyExcel.GetType().GetProperties().ToList();
  240. // 检查单元格是否在合并区域内
  241. bool isMerged = false; bool isNewMergedRegion = false;
  242. CellRangeAddress currentMergedRegion = null;
  243. for (int i = 0; i < sheet.NumMergedRegions; i++)
  244. {
  245. CellRangeAddress mergedRegion = sheet.GetMergedRegion(i);
  246. if (mergedRegion.FirstRow <= r && mergedRegion.LastRow >= r && mergedRegion.FirstColumn <= j && mergedRegion.LastColumn >= j)
  247. {
  248. isMerged = true;
  249. currentMergedRegion = mergedRegion;
  250. break;
  251. }
  252. }
  253. if (isMerged && r > 1)
  254. {
  255. // 判断是否为新的合并区域
  256. isNewMergedRegion = !processedMergedRegions.Contains(currentMergedRegion);
  257. if (isNewMergedRegion)
  258. {
  259. // 若为新的合并区域,添加到已处理列表
  260. processedMergedRegions.Add(currentMergedRegion);
  261. }
  262. }
  263. foreach (var x in properties)
  264. {
  265. if (x.IsDefined(typeof(DescriptionAttribute)))
  266. {
  267. //如果是合并行且不是第一列(第一列要拿数据的) cell.IsMergedCell
  268. if (isMerged && r > 1)
  269. {
  270. var value = x.GetValue(copyExcel)?.ToString();
  271. //新行不是空且与旧值不想等,则重新赋值.合并行后当前行数据是空.
  272. if ((!string.IsNullOrEmpty(cellValue) && cellValue != value) || isNewMergedRegion)
  273. {
  274. var attribute = x.GetCustomAttribute<DescriptionAttribute>();
  275. if (attribute.Description == headrow4.GetCell(j).ToString())
  276. {
  277. x.SetValue(copyExcel, cellValue);
  278. researchExcel = copyExcel;
  279. continue;
  280. }
  281. }
  282. }
  283. else
  284. {
  285. var attribute = x.GetCustomAttribute<DescriptionAttribute>();
  286. if (attribute.Description == headrow4.GetCell(j).ToString())
  287. {
  288. x.SetValue(copyExcel, cellValue);
  289. continue;
  290. }
  291. }
  292. }
  293. }
  294. }
  295. researchExcel = copyExcel;
  296. if (string.IsNullOrEmpty(copyExcel.Column4))
  297. {
  298. throw new Exception("巡检内容不能为空");
  299. }
  300. var copyExcel0 = copyExcel.Copy();
  301. // 检查 Column0 是否已存在
  302. var IsExist = researchExcels.Where(excel => excel.Column0.StartsWith(copyExcel.Column0)).ToList();
  303. if (IsExist != null && IsExist.Count > 0)
  304. {
  305. copyExcel0.Column0 += "-" + IsExist.Count;
  306. }
  307. researchExcels.Add(copyExcel0);
  308. }
  309. var content = researchExcels.Select(x => new { x.Column0, x.Column4, x.Column5 }).Distinct().ToList();
  310. var contentGroup = researchExcels.Select(x => new { x.Column1, x.Column2, x.Column3, x.Column6 }).Distinct().ToList();
  311. var contentList = await _tispContentService.GetAllAsync(storeCode);
  312. TispContentGroupsSearchModel searchModel = new TispContentGroupsSearchModel();
  313. searchModel.C_StoreCode = storeCode;
  314. var contentGroupList = await _tispContentGroupService.GetContentGroupsAsync(searchModel);
  315. List<TISP_Content> contentArray = new List<TISP_Content>();
  316. List<TISP_ContentGroup> contentGroupArray = new List<TISP_ContentGroup>();
  317. List<TISP_ContentGroupItem> groupItemArray = new List<TISP_ContentGroupItem>();
  318. foreach (var item in content)
  319. {
  320. int sort = 0;
  321. if (!string.IsNullOrEmpty(item.Column0))
  322. {
  323. sort = UtilConvert.GetIntByStr(item.Column0);
  324. }
  325. var contentModel = contentList.Where(c => c.C_Name == item.Column4).FirstOrDefault();
  326. if (contentModel == null)
  327. {
  328. TISP_Content contentAdd = new TISP_Content
  329. {
  330. C_ID = Guid.NewGuid(),
  331. C_StoreCode = storeCode,
  332. C_Name = item.Column4,
  333. C_AlarmLevel = "1",//1=轻;2 = 中;3 = 重
  334. C_Number = item.Column0,
  335. I_Sort = sort,
  336. C_Remark = item.Column5,
  337. C_CreateBy = _claims.ApiUserId,
  338. D_CreateOn = DateTime.Now,
  339. C_Status = "1",
  340. };
  341. contentArray.Add(contentAdd);
  342. }
  343. }
  344. int sortNum = 0;
  345. foreach (var item in contentGroup)
  346. {
  347. int type = 1;
  348. //1=防火检查记录表(周);2 = 隐患整改验收单;3 = 消防设施每周检查记录;4 = 防火巡查记录表
  349. switch (item.Column1)
  350. {
  351. case "防火检查记录表(周)": type = 1; break;
  352. case "隐患整改验收单": type = 2; break;
  353. case "消防设施每周检查记录": type = 3; break;
  354. case "防火巡查记录表": type = 4; break;
  355. case "点检工单记录表": type = 5; break;
  356. }
  357. var GroupModel = contentGroupList.Where(g => g.C_Name == item.Column2 && g.I_Type == type && g.C_PatrolItem == item.Column3).FirstOrDefault();
  358. TISP_ContentGroup contentGroupAdd = null;
  359. if (GroupModel == null)
  360. {
  361. contentGroupAdd = new TISP_ContentGroup
  362. {
  363. G_ID = Guid.NewGuid(),
  364. C_StoreCode = storeCode,
  365. C_DevStoreCode= devCode,
  366. C_Name = item.Column2,
  367. I_Sort = sortNum,
  368. C_Remark = item.Column6,
  369. I_Type = type,
  370. C_PatrolItem = item.Column3,
  371. G_CreateBy = _claims.ApiUserId,
  372. D_CreateOn = DateTime.Now,
  373. C_Status = "1",
  374. };
  375. contentGroupArray.Add(contentGroupAdd);
  376. sortNum++;
  377. }
  378. }
  379. foreach (var item in researchExcels)
  380. {
  381. int type = 1;
  382. //1=防火检查记录表(周);2 = 隐患整改验收单;3 = 消防设施每周检查记录;4 = 防火巡查记录表
  383. switch (item.Column1)
  384. {
  385. case "防火检查记录表(周)": type = 1; break;
  386. case "隐患整改验收单": type = 2; break;
  387. case "消防设施每周检查记录": type = 3; break;
  388. case "防火巡查记录表": type = 4; break;
  389. case "点检工单记录表": type = 5; break;
  390. }
  391. Guid contentGroupId = new Guid(), contentId = new Guid();
  392. if (contentArray.Count > 0)
  393. {
  394. var ccmodel = contentArray.Where(c => c.C_Name == item.Column4).FirstOrDefault();
  395. if (ccmodel != null) { contentId = ccmodel.C_ID; }
  396. }
  397. if (contentGroupArray.Count > 0)
  398. {
  399. var cgModel = contentGroupArray.Where(g => g.C_Name == item.Column2 && g.I_Type == type && g.C_PatrolItem == item.Column3).FirstOrDefault();
  400. if (cgModel!=null) { contentGroupId = cgModel.G_ID; }
  401. }
  402. if (Guid.Empty == contentId && contentList.Count() > 0)
  403. {
  404. var ccModel = contentList.Where(c => c.C_Name == item.Column4).FirstOrDefault();
  405. if (ccModel != null) { contentId = ccModel.C_ID; }
  406. }
  407. if (Guid.Empty == contentGroupId && contentGroupList.Count() > 0)
  408. {
  409. var cgModel= contentGroupList.Where(g => g.C_Name == item.Column2 && g.I_Type == type && g.C_PatrolItem == item.Column3).FirstOrDefault();
  410. if (cgModel != null) { contentGroupId = cgModel.G_ID; }
  411. }
  412. if (Guid.Empty != contentId && Guid.Empty != contentGroupId)
  413. {
  414. var ItemList = await _tispContentGroupItemService.GetContentGroupItemByIdAsync(contentGroupId, contentId);
  415. if (ItemList == null || ItemList.Count() == 0)
  416. {
  417. var groupItemAdd = new TISP_ContentGroupItem
  418. {
  419. G_ID = Guid.NewGuid(),
  420. G_ContentGroupCode = contentGroupId,
  421. G_ContentCode = contentId
  422. };
  423. groupItemArray.Add(groupItemAdd);
  424. }
  425. }
  426. }
  427. if (contentArray.Count == 0 && contentGroupArray.Count == 0 && groupItemArray.Count == 0)
  428. {
  429. return new ApiResult(ReturnCode.GeneralError, "要导入的数据已经存在!");
  430. }
  431. else
  432. {
  433. bool result = await _tispContentService.CreateIspContent(contentArray, contentGroupArray, groupItemArray);
  434. if (result)
  435. {
  436. return new ApiResult(ReturnCode.Success);
  437. }
  438. else
  439. {
  440. return new ApiResult(ReturnCode.GeneralError);
  441. }
  442. }
  443. }
  444. catch (Exception ex)
  445. {
  446. return new ApiResult(ReturnCode.GeneralError,ex.Message);
  447. }
  448. }
  449. }
  450. }