EmrServer.java 39 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051
  1. package thyyxxk.webserver.service.zhuyuanyisheng.emr;
  2. import cn.hutool.core.codec.Base64;
  3. import cn.hutool.core.convert.Convert;
  4. import cn.hutool.core.util.StrUtil;
  5. import cn.hutool.extra.spring.SpringUtil;
  6. import cn.hutool.http.HttpRequest;
  7. import com.alibaba.fastjson.JSON;
  8. import com.alibaba.fastjson.JSONArray;
  9. import com.alibaba.fastjson.JSONObject;
  10. import com.alibaba.fastjson.serializer.SerializerFeature;
  11. import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
  12. import com.dtflys.forest.Forest;
  13. import com.dtflys.forest.http.ForestRequest;
  14. import com.dtflys.forest.interceptor.Interceptor;
  15. import com.dtflys.forest.reflection.ForestMethod;
  16. import com.dtflys.forest.utils.ForestDataType;
  17. import lombok.RequiredArgsConstructor;
  18. import lombok.extern.slf4j.Slf4j;
  19. import org.apache.commons.collections4.ListUtils;
  20. import org.jetbrains.annotations.NotNull;
  21. import org.springframework.beans.factory.annotation.Value;
  22. import org.springframework.scheduling.annotation.Scheduled;
  23. import org.springframework.stereotype.Service;
  24. import org.springframework.transaction.annotation.Transactional;
  25. import thyyxxk.webserver.config.envionment.ApiUrl;
  26. import thyyxxk.webserver.config.envionment.ArchiveConfig;
  27. import thyyxxk.webserver.config.envionment.SystemConfig;
  28. import thyyxxk.webserver.config.exception.ExceptionEnum;
  29. import thyyxxk.webserver.constants.Capacity;
  30. import thyyxxk.webserver.constants.EmrType;
  31. import thyyxxk.webserver.dao.his.LoginDao;
  32. import thyyxxk.webserver.dao.his.zhuyuanyisheng.emr.EmrFolderDao;
  33. import thyyxxk.webserver.dao.his.zhuyuanyisheng.emr.EmrPatientDao;
  34. import thyyxxk.webserver.entity.ResultVo;
  35. import thyyxxk.webserver.entity.datamodify.YzTemperature;
  36. import thyyxxk.webserver.entity.dictionary.CodeName;
  37. import thyyxxk.webserver.entity.drg.AuxiliaryFillingOfDiagnosis;
  38. import thyyxxk.webserver.entity.fluorescenceTest.FluorescenceSpecimenResult;
  39. import thyyxxk.webserver.entity.login.UserInfo;
  40. import thyyxxk.webserver.entity.medicalinsurance.inpatient.ZyPatientInfo;
  41. import thyyxxk.webserver.entity.outpatient.thmz.MzEmrPatientData;
  42. import thyyxxk.webserver.entity.zhuyuanyisheng.ZyZkList;
  43. import thyyxxk.webserver.entity.zhuyuanyisheng.emr.*;
  44. import thyyxxk.webserver.entity.zhuyuanyisheng.jianyanjiancha.YshYjReq;
  45. import thyyxxk.webserver.entity.zhuyuanyisheng.shoushu.OpRecord;
  46. import thyyxxk.webserver.http.drg.DrgWebApi;
  47. import thyyxxk.webserver.service.PublicServer;
  48. import thyyxxk.webserver.service.hutoolcache.ExtraCache;
  49. import thyyxxk.webserver.service.externalhttp.emr.EmrEditor;
  50. import thyyxxk.webserver.service.hutoolcache.UserCache;
  51. import thyyxxk.webserver.utils.*;
  52. import javax.annotation.PostConstruct;
  53. import java.util.*;
  54. import java.util.stream.Collectors;
  55. @Service
  56. @Slf4j
  57. @RequiredArgsConstructor
  58. public class EmrServer {
  59. private final EmrPatientDao dao;
  60. private final EmrEditor emr;
  61. private final EmrFolderDao folderDao;
  62. private final PublicServer publicServer;
  63. private final LoginDao loginDao;
  64. private final ExtraCache extraCache;
  65. private final UserCache userCache;
  66. private static final ArchiveConfig archiveConfig = SpringUtil.getBean(ArchiveConfig.class);
  67. private static String emrToken = null;
  68. private static String EMR_URL;
  69. public static final String THIS_IS_DIR = "This is Dir";
  70. private final DrgWebApi drgWebApi;
  71. private final ApiUrl apiUrl;
  72. private final SystemConfig systemConfig;
  73. @Value("${forest.variables.emrUrl}")
  74. public void setUrl(String key) {
  75. EMR_URL = key;
  76. }
  77. public static class EmrInterceptor implements Interceptor<String> {
  78. @Override
  79. public void onInvokeMethod(ForestRequest request, ForestMethod method, Object[] args) {
  80. request.addHeader("emr-token", emrToken);
  81. }
  82. }
  83. @PostConstruct
  84. @Scheduled(cron = "* * */12 * * ?")
  85. public void token() {
  86. if (EMR_URL == null) {
  87. return;
  88. }
  89. try {
  90. String URL = EMR_URL.replace("/emr/runtime/api/v1", "");
  91. String result = HttpRequest.post(URL + "/emr/runtime/api/v1/oauth/token")
  92. .header("Authorization", "Basic " + Base64.encode("user:dc71ccfec05b799ad52360c48d504019"))
  93. .form("grant_type", "client_credentials").execute().body();
  94. JSONObject data = JSONObject.parseObject(result);
  95. emrToken = data.getString("access_token");
  96. } catch (Exception ignored) {
  97. }
  98. }
  99. /**
  100. * 获取这个患者这个住院次数的全部电子病历
  101. *
  102. * @param patNo 住院号
  103. * @param times 住院次数
  104. * @return 数据
  105. */
  106. public ResultVo<List<EmrPatientData>> getPatientDataTree(String patNo, Integer times) {
  107. List<EmrPatientData> list = dao.getPatientData(patNo, times);
  108. List<EmrPatientData> folder = extraCache.getValue("电子病历文件夹", () -> folderDao.getFolderByName("HOSPITAL"));
  109. folder = JSONObject.parseArray(JSONObject.toJSONString(folder), EmrPatientData.class);
  110. if (ListUtil.isBlank(list)) {
  111. return ResultVoUtil.success(folder);
  112. }
  113. folder.addAll(list);
  114. Map<String, EmrPatientData> folderMap = new HashMap<>();
  115. List<EmrPatientData> result = new ArrayList<>();
  116. folder.add(EmrPatientData.builder()
  117. .id(null)
  118. .emrDocumentId("noFolder")
  119. .name("可拖动病历调整到合适文件夹,右键确认排序")
  120. .emrCategoryCode(THIS_IS_DIR)
  121. .sort(0)
  122. .children(new ArrayList<>())
  123. .parent(null)
  124. .build());
  125. for (EmrPatientData item : folder) {
  126. folderMap.put(item.getEmrDocumentId(), item);
  127. if (item.getParent() == null) {
  128. if ("group-category".equals(item.getType()))
  129. result.add(item);
  130. }
  131. }
  132. for (EmrPatientData item : folder) {
  133. String key = item.getParent();
  134. EmrPatientData parent = folderMap.get(key);
  135. if (parent != null) {
  136. if (parent.getChildren() == null) {
  137. parent.setChildren(new ArrayList<>());
  138. }
  139. parent.getChildren().add(item);
  140. } else {
  141. if (THIS_IS_DIR.equals(item.getEmrCategoryCode())) {
  142. continue;
  143. }
  144. EmrPatientData no = folderMap.get("noFolder");
  145. no.getChildren().add(item);
  146. }
  147. }
  148. result.sort(Comparator.comparing((o) -> o.getSort() == null ? 0 : o.getSort()));
  149. return ResultVoUtil.success(result);
  150. }
  151. public ResultVo<Map<String, List<JSONObject>>> getEmrTree(String type) {
  152. Map<String, List<JSONObject>> map = new HashMap<>(Capacity.TWO);
  153. if ("all".equals(type)) {
  154. map.put("all", wholeHospitalTemplate());
  155. map.put("dept", getDeptTemplate());
  156. } else if ("dept".equals(type)) {
  157. map.put("dept", getDeptTemplate());
  158. } else if ("hosp".equals(type)) {
  159. map.put("all", wholeHospitalTemplate());
  160. }
  161. return ResultVoUtil.success(map);
  162. }
  163. private List<JSONObject> getDeptTemplate() {
  164. UserInfo user = userCache.getUserInfoByToken();
  165. JSONArray data = new JSONArray();
  166. if (ListUtil.notBlank(user.getPartTimeDept())) {
  167. user.getPartTimeDept().forEach(item -> {
  168. data.addAll(emr.getDeptList(item));
  169. });
  170. }
  171. data.addAll(emr.getDeptList(user.getDeptCode()));
  172. return emrToTree(data);
  173. }
  174. /**
  175. * 获取 整个医院模板
  176. */
  177. private List<JSONObject> wholeHospitalTemplate() {
  178. JSONArray data;
  179. if (StrUtil.isBlank(systemConfig.getEmrZyDeptCode())) {
  180. data = emr.getEmrTree();
  181. } else {
  182. data = emr.getDeptList(systemConfig.getEmrZyDeptCode());
  183. }
  184. return emrToTree(data);
  185. }
  186. /**
  187. * 电子病历转成树状图 二次封装
  188. *
  189. * @param data 模板数据
  190. * @return 返回
  191. */
  192. public List<JSONObject> emrToTree(JSONArray data) {
  193. List<JSONObject> tree = new ArrayList<>();
  194. if (data.isEmpty()) {
  195. return tree;
  196. }
  197. emrArrToTree(data, tree);
  198. return tree;
  199. }
  200. /**
  201. * 电子病历转成树状图
  202. *
  203. * @param data 电子病历
  204. * @param tree 树
  205. */
  206. private static void emrArrToTree(JSONArray data, List<JSONObject> tree) {
  207. Map<String, JSONObject> map = new HashMap<>(data.size());
  208. for (int i = 0; i < data.size(); i++) {
  209. JSONObject item = data.getJSONObject(i);
  210. if (item.get("parent") == null) {
  211. tree.add(item);
  212. }
  213. map.put(item.getString("_id"), item);
  214. }
  215. for (int i = 0; i < data.size(); i++) {
  216. JSONObject item = data.getJSONObject(i);
  217. if (map.containsKey(item.getString("parent"))) {
  218. JSONObject parentList = map.get(item.getString("parent"));
  219. parentList.computeIfAbsent("children", k -> new JSONArray());
  220. parentList.getJSONArray("children").add(item);
  221. }
  222. }
  223. }
  224. /**
  225. * 获取电子病历的片段
  226. *
  227. * @return 树
  228. */
  229. public ResultVo<List<JSONObject>> getSnippetTree() {
  230. JSONArray data = emr.getSnippetTree();
  231. List<JSONObject> tree = new ArrayList<>();
  232. emrArrToTree(data, tree);
  233. return ResultVoUtil.success(tree);
  234. }
  235. /**
  236. * 提取患者数据元
  237. *
  238. * @param param 结构化文档
  239. */
  240. public Map<String, Object> extractDataElement(EmrPatientData param) {
  241. EmrDataExtract emrDataExtract = dao.extractDataSource(param.getEmrCategoryCode());
  242. if (emrDataExtract == null) {
  243. return new HashMap<>();
  244. }
  245. // 获取 需要提取的数据元
  246. List<String> strings = JSON.parseArray(emrDataExtract.getDataExtract(), String.class);
  247. // 获取前端传入的数据元
  248. Map<String, Object> extractedData = getStringObjectMap(param, strings);
  249. EmrDataElement emrDataElement = dao.obtainPatientSOriginalData(param.getPatNo(), param.getTimes());
  250. if (emrDataElement == null) {
  251. // 创建的病历可以解锁了
  252. dao.insertDataSource(param.getPatNo(), param.getTimes(), JSON.toJSONString(extractedData));
  253. } else {
  254. JSONObject jsonObject = JSONObject.parseObject(emrDataElement.getDataElement());
  255. jsonObject.putAll(extractedData);
  256. dao.updatePatientDataSource(param.getPatNo(), param.getTimes(), JSON.toJSONString(jsonObject));
  257. }
  258. return extractedData;
  259. }
  260. @NotNull
  261. private static Map<String, Object> getStringObjectMap(EmrPatientData param, List<String> strings) {
  262. JSONObject elementData = param.getEmrDataElement();
  263. // 提取到的数据
  264. Map<String, Object> extractedData = new HashMap<>(strings.size());
  265. for (String el : strings) {
  266. if (elementData.containsKey(el)) {
  267. JSONObject item;
  268. try {
  269. item = elementData.getJSONObject(el);
  270. } catch (Exception ignore) {
  271. throw new RuntimeException("数据元提取失败,但病历保存成功。数据元名称:【" + el + "】,粘贴时请使用(文本粘贴)。");
  272. }
  273. Object value = item.get("value");
  274. if (value != null) {
  275. extractedData.put(el, value);
  276. }
  277. }
  278. }
  279. return extractedData;
  280. }
  281. /**
  282. * 保存电子病历
  283. *
  284. * @param param 参数
  285. * @return 提示
  286. */
  287. @Transactional(rollbackFor = Exception.class)
  288. public ResultVo<Map<String, Object>> insertEmrData(EmrPatientData param) {
  289. EmrPatientData patientData = dao.getCategoryCodeByDocumentId(param.getEmrDocumentId());
  290. // TM的新增病历也是用的这个方法,所以要考虑没有id的病历
  291. if (patientData != null) {
  292. if (!SignCompleteEnum.INIT.getCode().equals(patientData.getSignComplete())) {
  293. return ResultVoUtil.fail(ExceptionEnum.LOGICAL_ERROR, "患者完成电子签名无法修改。");
  294. }
  295. if (patientData.getDelFlag() == 1) {
  296. return ResultVoUtil.fail(ExceptionEnum.LOGICAL_ERROR, "病历已删除。");
  297. }
  298. }
  299. JSONObject saveJson = new JSONObject();
  300. saveJson.put("document", param.getDocumentData());
  301. try {
  302. emr.saveDocument(saveJson);
  303. } catch (Exception e) {
  304. return ResultVoUtil.fail(ExceptionEnum.EMR_SAVE, "病历保存错误,请重试!" + e.getMessage());
  305. }
  306. param.setCreateId(TokenUtil.getInstance().getTokenUserId());
  307. Integer isUpdated = dao.whetherThereIsAMedicalRecord(param.getEmrDocumentId());
  308. if (isUpdated != null) {
  309. dao.updateCreatedTemplate(param);
  310. insertAFragmentOfTheCourseOfTheDisease(isUpdated, param.getFragment());
  311. } else {
  312. // 新增的时候插入数据
  313. ZyPatientInfo patInfo = publicServer.getPatInfo(param.getPatNo(), param.getTimes());
  314. param.setReferPhysician(patInfo.getReferPhysician());
  315. param.setConsultPhysician(patInfo.getConsultPhysician());
  316. param.setDeptDirector(patInfo.getDeptDirector());
  317. dao.emrInsertForTheFirstTime(param);
  318. insertAFragmentOfTheCourseOfTheDisease(param.getId(), param.getFragment());
  319. }
  320. Map<String, Object> extractedData;
  321. try {
  322. extractedData = extractDataElement(param);
  323. } catch (Exception e) {
  324. return ResultVoUtil.fail(ExceptionEnum.EMR_EXTRACT_OBJECTS, e.getMessage());
  325. }
  326. return ResultVoUtil.success(extractedData);
  327. }
  328. /**
  329. * 插入病程记录的片段
  330. *
  331. * @param id 病程记录
  332. * @param fragment 片段
  333. */
  334. private void insertAFragmentOfTheCourseOfTheDisease(Integer id, List<EmrProgressNote> fragment) {
  335. if (ListUtil.isBlank(fragment)) {
  336. return;
  337. }
  338. dao.delOldFragment(id);
  339. List<List<EmrProgressNote>> lists = ListUtils.partition(fragment, 50);
  340. for (List<EmrProgressNote> list : lists) {
  341. dao.insertFragment(list, id);
  342. }
  343. }
  344. /**
  345. * 根据患者的电子病历 id 来删除
  346. * 这里用的是逻辑删除
  347. *
  348. * @param documentId 电子病历id
  349. * @return 删除电子病历的同时需要删除, 提取到的数据源.
  350. */
  351. public ResultVo<List<String>> deletePatientEmrByDocumentId(String documentId) {
  352. EmrPatientData patientData = dao.getCategoryCodeByDocumentId(documentId);
  353. if (!SignCompleteEnum.INIT.getCode().equals(patientData.getSignComplete())) {
  354. return ResultVoUtil.fail(ExceptionEnum.LOGICAL_ERROR, "患者完成电子签名无法删除。");
  355. }
  356. // 医务部可以自由删除其他医生的病历
  357. if (!publicServer.needRule(38)) {
  358. if (!patientData.getCreateId().equals(TokenUtil.getInstance().getTokenUserId())) {
  359. return ResultVoUtil.fail(ExceptionEnum.LOGICAL_ERROR, "创建人不是你无法删除。");
  360. }
  361. }
  362. EmrDataExtract emrDataExtract = dao.extractDataSource(patientData.getEmrCategoryCode());
  363. List<String> strings = new ArrayList<>();
  364. if (emrDataExtract != null) {
  365. // 获取 需要提取的数据元
  366. strings = JSON.parseArray(emrDataExtract.getDataExtract(), String.class);
  367. // 获取到已经提取的数据
  368. EmrDataElement emrDataElement = dao.obtainPatientSOriginalData(patientData.getPatNo(), patientData.getTimes());
  369. // 转 json
  370. JSONObject jsonObject = JSONObject.parseObject(emrDataElement.getDataElement());
  371. for (String string : strings) {
  372. // 删除提取到的数据
  373. jsonObject.remove(string);
  374. }
  375. // 更新患者数据元
  376. dao.updatePatientDataSource(patientData.getPatNo(), patientData.getTimes(), JSON.toJSONString(jsonObject));
  377. }
  378. // 删除病历
  379. dao.deletePatientEmrByDocumentId(documentId, TokenUtil.getInstance().getTokenUserId());
  380. return ResultVoUtil.success(strings);
  381. }
  382. /**
  383. * 查询患者是否有指定的病历了
  384. *
  385. * @param param 住院号,住院次数,病历编码
  386. * @return 返回 Boolean
  387. */
  388. public ResultVo<Boolean> queryWhetherThePatientHasASpecifiedMedicalRecord(EmrPatientData param) {
  389. return ResultVoUtil.success(dao.queryWhetherThePatientHasASpecifiedMedicalRecord(param.getPatNo(), param.getTimes(), param.getEmrCategoryCode()) == 1);
  390. }
  391. /**
  392. * 医生诊断热缩
  393. *
  394. * @param userCode 医生编码
  395. * @param code 诊断 code
  396. * @param tableName 表名
  397. */
  398. public void hotSearchSorting(String userCode, String code, String tableName) {
  399. boolean insert = dao.queryWhetherThereIsHotSearch(userCode, code, tableName) == 0;
  400. if (insert) {
  401. dao.hotSearchSorting(userCode, code, tableName);
  402. } else {
  403. dao.updateHotSearch(userCode, code, tableName);
  404. }
  405. }
  406. /**
  407. * 根据文档 id 提交病历
  408. *
  409. * @param documentId 文档id
  410. * @return 返回数据
  411. */
  412. public ResultVo<String> submitMedicalRecord(String documentId) {
  413. String userCode = TokenUtil.getInstance().getTokenUserId();
  414. String createdId = dao.getDocumentIdCreatedBy(documentId);
  415. if (userCode.equals(createdId)) {
  416. dao.updateSubmissionFlag(documentId, userCode);
  417. return ResultVoUtil.success(ExceptionEnum.SUCCESS_AND_EL_MESSAGE, "提交成功");
  418. }
  419. return ResultVoUtil.fail(ExceptionEnum.LOGICAL_ERROR, "创建人不一致无法修改。");
  420. }
  421. public ResultVo<String> audit(String documentId) {
  422. String userCode = TokenUtil.getInstance().getTokenUserId();
  423. int count = dao.updateAuditByDocumentId(documentId, userCode);
  424. if (count > 0) {
  425. return ResultVoUtil.success(ExceptionEnum.SUCCESS_AND_EL_MESSAGE, "审核成功");
  426. }
  427. return ResultVoUtil.success(ExceptionEnum.NO_DATA_EXIST, "操作失败,可能原因,已经审核完成或还未保存。");
  428. }
  429. /**
  430. * 获取电子病历数据
  431. *
  432. * @param panNo 住院号
  433. * @param times 住院次数
  434. * @param etType 类型
  435. * @return 返回数据
  436. */
  437. public JSONArray getEmrData(String panNo, Integer times, EmrType etType) {
  438. // 查询这个病历的 唯一 id
  439. List<String> idList = dao.getDocumentIdByPatietn(panNo, times, etType.getName());
  440. JSONArray returnArray = new JSONArray();
  441. if (ListUtil.isBlank(idList)) {
  442. return returnArray;
  443. }
  444. Map<String, JSONObject> surgicalSequencing = new TreeMap<>();
  445. for (String s : idList) {
  446. JSONObject jsonObject = emr.getEditorJsonDataByDocumentId(s);
  447. if (etType.getCode() == EmrType.DIAGNOSIS.getCode()) {
  448. returnArray.addAll(extractDiagnosis(jsonObject));
  449. } else if (etType.getCode() == EmrType.OPERATION.getCode()) {
  450. String key = DateUtil.formatDatetime(jsonObject.getJSONObject("手术日期").getDate("value"));
  451. surgicalSequencing.put(key, jsonObject);
  452. }
  453. }
  454. if (!surgicalSequencing.isEmpty()) {
  455. for (Map.Entry<String, JSONObject> item : surgicalSequencing.entrySet()) {
  456. returnArray.add(item.getValue());
  457. }
  458. }
  459. return returnArray;
  460. }
  461. /**
  462. * 获取患者提取到的数据
  463. *
  464. * @param patNo 住院号
  465. * @param times 次数
  466. * @return 返回 json
  467. */
  468. public JSONObject getEmrPatientData(String patNo, Integer times) {
  469. String js = dao.getEmrPatientData(patNo, times);
  470. if (StringUtil.isBlank(js)) {
  471. return new JSONObject();
  472. }
  473. return JSON.parseObject(js);
  474. }
  475. /**
  476. * 获取患者的手术
  477. *
  478. * @param patNo 获取住院号
  479. * @param times 获取住院次数
  480. * @return 返回数据
  481. */
  482. public JSONArray getPatientSurgery(String patNo, Integer times) {
  483. List<String> idList = dao.getDocumentIdByPatietn(patNo, times, "shoushujilu");
  484. JSONArray returnArray = new JSONArray();
  485. if (ListUtil.isBlank(idList)) {
  486. return returnArray;
  487. }
  488. Map<String, JSONObject> surgicalSequencing = new TreeMap<>();
  489. for (String s : idList) {
  490. JSONObject jsonObject = emr.getEditorJsonDataByDocumentId(s);
  491. JSONObject surgeryDate = jsonObject.getJSONObject("手术日期");
  492. if (null == surgeryDate) {
  493. continue;
  494. }
  495. String key = DateUtil.formatDatetime(surgeryDate.getDate("value"));
  496. if (StringUtil.isBlank(key)) {
  497. continue;
  498. }
  499. surgicalSequencing.put(key, jsonObject);
  500. }
  501. if (!surgicalSequencing.isEmpty()) {
  502. for (Map.Entry<String, JSONObject> item : surgicalSequencing.entrySet()) {
  503. returnArray.add(item.getValue());
  504. }
  505. }
  506. return returnArray;
  507. }
  508. private JSONArray extractDiagnosis(JSONObject jsonObject) {
  509. JSONObject diagnosis = jsonObject.getJSONObject("入院诊断");
  510. if (diagnosis != null) {
  511. JSONArray diagnosisArr = diagnosis.getJSONArray("value");
  512. if (diagnosisArr != null) {
  513. return diagnosisArr;
  514. }
  515. }
  516. return new JSONArray();
  517. }
  518. public ResultVo<String> getDrgIntelligentGrouping(String patNo, Integer times) {
  519. if (StrUtil.isBlank(apiUrl.getDrgWebApi())) {
  520. return ResultVoUtil.fail(ExceptionEnum.LOGICAL_ERROR, "未找到DRG服务");
  521. }
  522. // 获取诊断
  523. JSONArray diagnosis = getEmrData(patNo, times, EmrType.DIAGNOSIS);
  524. // 获取手术
  525. JSONArray operation = getEmrData(patNo, times, EmrType.OPERATION);
  526. int ledgerSn = publicServer.getLedgerSn(patNo, times);
  527. // 获取患者数据
  528. Map<String, String> mapData = dao.drgPatientInfo(patNo, times, ledgerSn);
  529. if (mapData == null) {
  530. return ResultVoUtil.fail(ExceptionEnum.LOGICAL_ERROR, "没有查询到患者信息。");
  531. }
  532. mapData.put("visit_id", patNo + "_" + times + "_" + ledgerSn);
  533. mapData.put("bah", patNo + "_" + times);
  534. for (int i = 0; i < diagnosis.size(); i++) {
  535. JSONObject item = diagnosis.getJSONObject(i);
  536. if (i == 0) {
  537. mapData.put("jbdm", item.getString("code"));
  538. mapData.put("zyzd", item.getString("name"));
  539. } else {
  540. mapData.put("jbdm" + i, item.getString("code"));
  541. }
  542. }
  543. int operationIndex = 1;
  544. for (int i = 0; i < operation.size(); i++) {
  545. JSONObject item = operation.getJSONObject(i);
  546. JSONArray list;
  547. try {
  548. list = item.getJSONObject("手术名称").getJSONArray("value");
  549. } catch (Exception e) {
  550. continue;
  551. }
  552. if (list == null) {
  553. continue;
  554. }
  555. for (int a = 0; a < list.size(); a++) {
  556. JSONObject valueList = list.getJSONObject(a);
  557. mapData.put("ssjczbm" + operationIndex, valueList.getString("code"));
  558. if (operationIndex == 1) {
  559. mapData.put("ssjczmc" + operationIndex, valueList.getString("name"));
  560. }
  561. operationIndex++;
  562. }
  563. }
  564. AuxiliaryFillingOfDiagnosis data = EntityCopy.Copy(mapData, AuxiliaryFillingOfDiagnosis.class);
  565. String toJsonStr = JSON.toJSONString(data, SerializerFeature.WriteNullStringAsEmpty);
  566. return drgWebApi.getDrgIntelligentGrouping(JSON.parseObject(toJsonStr));
  567. }
  568. /**
  569. * 获取患者最大的住院次数
  570. *
  571. * @param patNo 住院号
  572. * @return 返回最大次数
  573. */
  574. public ResultVo<Integer> getDischargeTimes(String patNo) {
  575. Integer times = dao.getMaxTimes(patNo);
  576. if (times == null) {
  577. return ResultVoUtil.fail(ExceptionEnum.LOGICAL_ERROR, "没有查询到患者信息。");
  578. }
  579. return ResultVoUtil.success(times);
  580. }
  581. public ResultVo<JSONObject> getExtractDataElement(String patNo, Integer times) {
  582. Map<String, String> map = dao.selectEmrDataElement(patNo, times);
  583. if (map == null) {
  584. return ResultVoUtil.success();
  585. } else {
  586. return ResultVoUtil.success(JSONObject.parseObject(map.get("data_element")));
  587. }
  588. }
  589. public ResultVo<List<CodeName>> getAllWards() {
  590. return ResultVoUtil.success(loginDao.getAllWards());
  591. }
  592. /**
  593. * 获取患者已经被删除的病历,管理员不需要限制
  594. *
  595. * @param patNo 住院号
  596. * @return 返回
  597. */
  598. public ResultVo<List<EmrPatientData>> getDeleteMedicalRecord(String patNo) {
  599. Integer times = dao.getAdmissTimes(patNo);
  600. if (times == null) {
  601. return ResultVoUtil.fail(ExceptionEnum.NULL_POINTER, "出院患者无法恢复病历,如果要恢复请先召回在院。");
  602. }
  603. String userCode = TokenUtil.getInstance().getTokenUserId();
  604. if (publicServer.needRule()) {
  605. userCode = "";
  606. }
  607. return ResultVoUtil.success(dao.selectEmrDeteles(patNo, times, userCode));
  608. }
  609. public ResultVo<JSONObject> getInvalidByDocumentId(String id) {
  610. return ResultVoUtil.success(emr.getInvalidByDocumentIdApi(id));
  611. }
  612. /**
  613. * 恢复电子病历
  614. *
  615. * @param documentId 电子病历文档
  616. * @return 返回提示
  617. */
  618. public ResultVo<String> resumeMedicalRecords(String documentId) {
  619. EmrPatientData data = dao.selectemPatientDataOne(documentId);
  620. if (data.getDelFlag() == 0) {
  621. return ResultVoUtil.fail(ExceptionEnum.NULL_POINTER, "病历已经被恢复了请勿重复点击");
  622. }
  623. Integer times = dao.getAdmissTimes(data.getPatNo());
  624. if (times == null) {
  625. return ResultVoUtil.fail(ExceptionEnum.NULL_POINTER, "出院患者无法恢复病历,如果要恢复请先召回在院。");
  626. }
  627. return ResultVoUtil.success(ExceptionEnum.SUCCESS_AND_EL_MESSAGE, "病历恢复成功。");
  628. }
  629. /**
  630. * 查询出院患者的申请
  631. *
  632. * @param flag 1 只查询自己的 2 查询审核的
  633. * @return 提示
  634. */
  635. public ResultVo<List<DischargedEmrModifyApply>> getDisReqEmr(Integer flag) {
  636. QueryWrapper<?> qw = new QueryWrapper<>();
  637. if (flag == 1) {
  638. qw.eq("req_id", TokenUtil.getInstance().getTokenUserId());
  639. } else if (flag == 2) {
  640. qw.eq("req_status", 0);
  641. }
  642. qw.ge("edit_date", DateUtil.formatDatetime(new Date(), DateUtil.DATE));
  643. return ResultVoUtil.success(dao.getDisReqEmr(qw));
  644. }
  645. public ResultVo<List<DischargedEmrModifyApply>> downloadTheDischargeEdit(String start, String end) {
  646. return ResultVoUtil.success(dao.selectDisReqEmyDownload(start, end));
  647. }
  648. /**
  649. * 申请出院患者修改
  650. *
  651. * @param param 参数
  652. * @return 提示
  653. */
  654. @Transactional(rollbackFor = Exception.class)
  655. public ResultVo<String> saveDisEmrReq(DischargedEmrModifyApply param) {
  656. param.setReqId(TokenUtil.getInstance().getTokenUserId());
  657. int num = dao.deleteRequest(param.getPatNo());
  658. dao.insertApplicationEmrDis(param);
  659. if (num > 0) {
  660. return ResultVoUtil.success(ExceptionEnum.SUCCESS_AND_EL_MESSAGE, "修改原申请成功。");
  661. } else {
  662. return ResultVoUtil.success(ExceptionEnum.SUCCESS_AND_EL_MESSAGE, "申请成功。");
  663. }
  664. }
  665. /**
  666. * 删除申请
  667. *
  668. * @param patNo 住院号
  669. * @return 提示
  670. */
  671. public ResultVo<String> deleteDisEmrDis(String patNo) {
  672. dao.deleteRequest(patNo);
  673. return ResultVoUtil.success(ExceptionEnum.SUCCESS_AND_EL_MESSAGE, "操作成功。");
  674. }
  675. /**
  676. * 通过申请
  677. *
  678. * @param patNo 住院号
  679. * @return 提示
  680. */
  681. public ResultVo<String> adoptEmrDisReq(String patNo) {
  682. int num = dao.adoptEmrDisReq(patNo, null, 1);
  683. if (num > 0) {
  684. return ResultVoUtil.success(ExceptionEnum.SUCCESS_AND_EL_MESSAGE);
  685. }
  686. return ResultVoUtil.fail(ExceptionEnum.INTERNAL_SERVER_ERROR, "操作失败,原申请可能已经被删除了。");
  687. }
  688. /**
  689. * 拒绝出院患者申请
  690. *
  691. * @param patNo 住院号
  692. * @param reviewNotes 审核信息
  693. * @return 提示
  694. */
  695. public ResultVo<String> refuseEmrDisReq(String patNo, String reviewNotes) {
  696. int num = dao.adoptEmrDisReq(patNo, reviewNotes, 2);
  697. if (num > 0) {
  698. return ResultVoUtil.success(ExceptionEnum.SUCCESS_AND_EL_MESSAGE);
  699. }
  700. return ResultVoUtil.fail(ExceptionEnum.INTERNAL_SERVER_ERROR, "操作失败,原申请可能已经被删除了。");
  701. }
  702. /**
  703. * 查询出院患者是否已经通过了编辑
  704. *
  705. * @param patNo 住院号
  706. * @return Boolean
  707. */
  708. public ResultVo<Boolean> isDisReqEdit(String patNo) {
  709. QueryWrapper<?> qw = new QueryWrapper<>();
  710. qw.eq("pat_no", patNo);
  711. qw.eq("req_status", 1);
  712. qw.ge("edit_date", DateUtil.formatDatetime(new Date(), DateUtil.DATE));
  713. List<DischargedEmrModifyApply> list = dao.getDisReqEmr(qw);
  714. if (ListUtil.isBlank(list)) {
  715. return ResultVoUtil.success(false);
  716. } else {
  717. return ResultVoUtil.success(true);
  718. }
  719. }
  720. public ResultVo<Object> getHistory(String documentId) {
  721. return ResultVoUtil.success(emr.getHistory(documentId));
  722. }
  723. public ResultVo<List<MedicalHistoryPrompts>> getEmrTips() {
  724. return ResultVoUtil.success(dao.selectEmrTips(TokenUtil.getInstance().getTokenUserId()));
  725. }
  726. public ResultVo<String> addEmrTips(MedicalHistoryPrompts param) {
  727. UserInfo userInfo = userCache.getUserInfoByToken();
  728. param.setDept(userInfo.getDeptCode());
  729. param.setCreatorId(userInfo.getCode());
  730. dao.insertEmrTips(param);
  731. return ResultVoUtil.success(ExceptionEnum.SUCCESS_AND_EL_MESSAGE);
  732. }
  733. public ResultVo<String> updateEmrTips(MedicalHistoryPrompts param) {
  734. if (param.getId() == null) {
  735. return ResultVoUtil.fail(ExceptionEnum.LOGICAL_ERROR, "id 为空无法更新。");
  736. }
  737. dao.updateEmrTips(param);
  738. return ResultVoUtil.success(ExceptionEnum.SUCCESS_AND_EL_MESSAGE);
  739. }
  740. public ResultVo<String> deleteEmrTips(Integer id) {
  741. dao.delEmrTips(id);
  742. return ResultVoUtil.success(ExceptionEnum.SUCCESS_AND_EL_MESSAGE);
  743. }
  744. /**
  745. * 在病程记录中获取文本数据
  746. *
  747. * @param name 名称
  748. * @return 返回数据
  749. */
  750. public ResultVo<List<MedicalHistoryPrompts>> getEmrTipsData(String name, int typeCode) {
  751. QueryWrapper<?> qw = new QueryWrapper<>();
  752. UserInfo us = userCache.getUserInfoByToken();
  753. if (typeCode == 1) {
  754. qw.eq("creator_id", us.getCode());
  755. } else if (typeCode == 2) {
  756. qw.eq("dept", us.getDeptCode());
  757. }
  758. qw.like("query_key", name)
  759. .eq("type_code", typeCode)
  760. .orderByDesc("count");
  761. return ResultVoUtil.success(dao.getEmrTipsData(qw));
  762. }
  763. public ResultVo<List<ZyPatientInfo>> getListOfDischargedPatients(String patNo) {
  764. List<ZyPatientInfo> list = dao.getListOfDischargedPatients(patNo);
  765. if (ListUtil.isBlank(list)) {
  766. return ResultVoUtil.fail(ExceptionEnum.NULL_POINTER, "没有" + patNo + "的出院记录。");
  767. }
  768. return ResultVoUtil.success(list);
  769. }
  770. public ResultVo<JSONObject> getYzTemperature(YzTemperatureParam param) {
  771. JSONObject js = new JSONObject();
  772. QueryWrapper<?> qw = new QueryWrapper<>();
  773. qw.eq("inpatient_no", param.getPatNo())
  774. .eq("admiss_times", param.getTimes())
  775. .ge("CONVERT(varchar(10), rec_date, 23)", param.getStartTime())
  776. .le("CONVERT(varchar(10), rec_date, 23)", param.getEndTime());
  777. List<YzTemperature> list = dao.getYzTemperature(qw);
  778. if (ListUtil.isBlank(list)) {
  779. return ResultVoUtil.success(js);
  780. }
  781. Map<Date, YzTemperature> map = new LinkedHashMap<>(list.size());
  782. for (YzTemperature item : list) {
  783. Date key = item.getTempDate();
  784. if (map.containsKey(key)) {
  785. YzTemperature yz = map.get(key);
  786. yz.setOtherInfo(nullToEmpty(yz.getOtherInfo()) + nullToEmpty(item.getOtherInfo()));
  787. } else {
  788. map.put(key, item);
  789. }
  790. }
  791. js.put("list", list);
  792. js.put("map", map);
  793. return ResultVoUtil.success(js);
  794. }
  795. public ResultVo<List<YshYjReq>> getExamine(String patNo, Integer times) {
  796. return ResultVoUtil.success(dao.getExamine(patNo, times));
  797. }
  798. private String nullToEmpty(String str) {
  799. if (StringUtil.isBlank(str)) {
  800. return "";
  801. }
  802. return str;
  803. }
  804. public ResultVo<ZyZkList> whetherItExistsInTheDepartment(String patNo, Integer times) {
  805. ZyZkList zk = dao.getMaxTransferDateByPatNo(patNo, times);
  806. if (zk != null) {
  807. zk.setTimeout(DateUtil.moreThanHours(zk.getOpTime(), 48));
  808. }
  809. return ResultVoUtil.success(zk);
  810. }
  811. public ResultVo<String> electronicMedicalRecordSequencing(List<EmrPatientData> data) {
  812. if (ListUtil.isBlank(data)) {
  813. return ResultVoUtil.fail(ExceptionEnum.LOGICAL_ERROR, "请选择有效的分类数据。");
  814. }
  815. ListUtil.batchList(data, EmrPatientDao.class, (m, item) -> {
  816. m.updateTheNode(item.getParent(), item.getId());
  817. });
  818. return ResultVoUtil.success(ExceptionEnum.SUCCESS_AND_EL_MESSAGE, "操作成功。");
  819. }
  820. public ResultVo<Map<String, List<EmrPatientData>>> getPastHistory(String patNo, Integer times) {
  821. List<EmrPatientData> list = dao.getHisPatInfo(patNo, times);
  822. if (ListUtil.isBlank(list)) {
  823. return ResultVoUtil.success(new HashMap<>());
  824. }
  825. Map<String, List<EmrPatientData>> grouping = list.stream().collect(
  826. Collectors.groupingBy(item -> item.getTimes().toString())
  827. );
  828. return ResultVoUtil.success(grouping);
  829. }
  830. public ResultVo<Map<Integer, List<MzEmrPatientData>>> getMzPatientDataByPatNo(String patNo) {
  831. String mzNo = dao.getMzNoByPatNo(patNo);
  832. Map<Integer, List<MzEmrPatientData>> map1 = new HashMap<>();
  833. if (StringUtil.isBlank(mzNo)) {
  834. return ResultVoUtil.success(map1);
  835. }
  836. List<MzEmrPatientData> mzList = dao.getEmrPatientMz(mzNo);
  837. if (ListUtil.isBlank(mzList)) {
  838. return ResultVoUtil.success(map1);
  839. }
  840. Map<Integer, List<MzEmrPatientData>> map = mzList.stream().collect(
  841. Collectors.groupingBy(MzEmrPatientData::getTimes)
  842. );
  843. return ResultVoUtil.success(map);
  844. }
  845. public ResultVo<List<FluorescenceSpecimenResult>> getFluorescenceSpecimenResult(String patNo, Integer times) {
  846. return ResultVoUtil.success(dao.selectFluorescenceSpecimenResult(patNo, times));
  847. }
  848. public ResultVo<List<OpRecord>> getOpRecordList(String patNo, Integer times) {
  849. return ResultVoUtil.success(dao.getOpRecordList(patNo, times));
  850. }
  851. public String getEmrToken() {
  852. return emrToken;
  853. }
  854. private static String createEmr(JSONObject param) {
  855. if (StrUtil.isBlank(archiveConfig.getArchiveUrl())) {
  856. throw new RuntimeException("电子病历生成路径为空");
  857. }
  858. param.put("emrToken", emrToken);
  859. String text = Forest.post(archiveConfig.getEmrArchiveUrl())
  860. .contentTypeJson()
  861. .addBody(param)
  862. .bodyType(ForestDataType.JSON)
  863. .addHeader("emr-token", emrToken)
  864. .execute(String.class);
  865. int index = text.lastIndexOf("<br/>buffer:,");
  866. if (index != -1) {
  867. String[] split = text.substring(text.lastIndexOf("<br/>buffer:,") + 13).split(",");
  868. return Base64.encode(Convert.toPrimitiveByteArray(split));
  869. }
  870. throw new RuntimeException(text);
  871. }
  872. public static String getPdfBase(String documentId) {
  873. JSONObject param = new JSONObject();
  874. param.put("emrToken", true);
  875. param.put("stream", true);
  876. param.put("arcive", false);
  877. param.put("patientId", false);
  878. param.put("params", new JSONObject() {{
  879. put("documentId", documentId);
  880. }});
  881. param.put("fileName", "病历pdf");
  882. param.put("type", "document");
  883. return createEmr(param);
  884. }
  885. public static String getPdfBase64ByTemplate(cn.hutool.json.JSONObject template, Object data) {
  886. JSONObject param = new JSONObject();
  887. param.put("emrToken", true);
  888. param.put("stream", true);
  889. param.put("arcive", false);
  890. param.put("patientId", false);
  891. param.put("params", new JSONObject() {{
  892. put("document", template);
  893. put("appContext", data);
  894. }});
  895. param.put("fileName", "病历pdf");
  896. param.put("type", "template");
  897. return createEmr(param);
  898. }
  899. public static String getPdfBase64ByCategoryId(String id, String categoryCode, Object data) {
  900. JSONObject param = new JSONObject();
  901. param.put("emrToken", true);
  902. param.put("stream", true);
  903. param.put("arcive", false);
  904. param.put("patientId", false);
  905. param.put("params", new JSONObject() {{
  906. put("categoryId", id);
  907. put("categoryCode", categoryCode);
  908. put("appContext", new JSONObject() {{
  909. put("data", data);
  910. }});
  911. }});
  912. param.put("fileName", "病历pdf");
  913. param.put("type", "template");
  914. return createEmr(param);
  915. }
  916. public ResultVo<String> newDir(EmrPatientData data) {
  917. data.setEmrDocumentId(SnowFlakeId.instance().nextId());
  918. data.setCreateDate(new Date());
  919. data.setCreateId(TokenUtil.getInstance().getTokenUserId());
  920. data.setEmrCategoryCode(THIS_IS_DIR);
  921. if (data.getParent() == null) {
  922. return ResultVoUtil.success(ExceptionEnum.LOGICAL_ERROR, "父节点为空请重新创建!");
  923. }
  924. dao.insert(data);
  925. return ResultVoUtil.success();
  926. }
  927. public ResultVo<String> delDir(String dirId) {
  928. boolean dirHasFile = dao.dirHasFile(dirId);
  929. if (dirHasFile) {
  930. return ResultVoUtil.fail(ExceptionEnum.LOGICAL_ERROR, "文件夹下面还存在文件无法删除。");
  931. }
  932. int i = dao.delDir(dirId, TokenUtil.getInstance().getTokenUserId());
  933. if (i > 0) {
  934. return ResultVoUtil.success(ExceptionEnum.SUCCESS_AND_EL_MESSAGE);
  935. }
  936. return ResultVoUtil.success(ExceptionEnum.LOGICAL_ERROR, "无法删除,创建人不是你,或者这个是文件。");
  937. }
  938. public ResultVo<String> rename(String name, String documentId) {
  939. int i = dao.rename(name, documentId);
  940. if (i > 0) {
  941. return ResultVoUtil.success(ExceptionEnum.SUCCESS_AND_EL_MESSAGE);
  942. }
  943. return ResultVoUtil.success(ExceptionEnum.LOGICAL_ERROR, "重命名失败。");
  944. }
  945. }