|  |  |  | 
|---|
|  |  |  | import cn.lihu.jh.framework.common.exception.ErrorCode; | 
|---|
|  |  |  | import cn.lihu.jh.framework.common.exception.enums.GlobalErrorCodeConstants; | 
|---|
|  |  |  | import cn.lihu.jh.framework.common.util.object.BeanUtils; | 
|---|
|  |  |  | import cn.lihu.jh.module.ecg.Utils; | 
|---|
|  |  |  | import cn.lihu.jh.module.ecg.controller.admin.room.vo.MonitorInfoVO; | 
|---|
|  |  |  | import cn.lihu.jh.module.ecg.dal.dataobject.checktype.CheckTypeDO; | 
|---|
|  |  |  | import cn.lihu.jh.module.ecg.dal.dataobject.queue.QueueDO; | 
|---|
|  |  |  | import cn.lihu.jh.module.ecg.dal.dataobject.queue.QueueStatisticDO; | 
|---|
|  |  |  | import cn.lihu.jh.module.ecg.dal.dataobject.room.RoomDO; | 
|---|
|  |  |  | import cn.lihu.jh.module.ecg.dal.mysql.call.CallMapper; | 
|---|
|  |  |  | import cn.lihu.jh.module.ecg.dal.mysql.checktype.CheckTypeMapper; | 
|---|
|  |  |  | import cn.lihu.jh.module.ecg.dal.mysql.devrent.DevRentMapper; | 
|---|
|  |  |  | import cn.lihu.jh.module.ecg.dal.mysql.queue.queueMapper2; | 
|---|
|  |  |  | import cn.lihu.jh.module.ecg.dal.mysql.queue.QueueMapper; | 
|---|
|  |  |  | import cn.lihu.jh.module.ecg.dal.mysql.room.RoomMapper; | 
|---|
|  |  |  | import cn.lihu.jh.module.ecg.enums.BedOpTypeEnum; | 
|---|
|  |  |  | import cn.lihu.jh.module.ecg.enums.BedStatusEnum; | 
|---|
|  |  |  | import cn.lihu.jh.module.ecg.enums.QueueStatusEnum; | 
|---|
|  |  |  | import cn.lihu.jh.module.infra.api.config.ConfigApi; | 
|---|
|  |  |  | import cn.lihu.jh.module.system.api.oauth2.OAuth2TokenApi; | 
|---|
|  |  |  | import lombok.extern.slf4j.Slf4j; | 
|---|
|  |  |  | import org.springframework.stereotype.Component; | 
|---|
|  |  |  | 
|---|
|  |  |  | public class QueueServiceTxFunctions { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Resource | 
|---|
|  |  |  | private ConfigApi configApi; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Resource | 
|---|
|  |  |  | private OAuth2TokenApi oAuth2TokenApi; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Resource | 
|---|
|  |  |  | private queueMapper2 queueMapper; | 
|---|
|  |  |  | private QueueMapper queueMapper; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | @Resource | 
|---|
|  |  |  | private RoomMapper roomMapper; | 
|---|
|  |  |  | 
|---|
|  |  |  | AtomicInteger openingFlag = new AtomicInteger(0); | 
|---|
|  |  |  | AtomicInteger curSeqNum = new AtomicInteger(0); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | ConcurrentHashMap<String, BedQueueBO > mapBedVsQueue = new ConcurrentHashMap<>(); | 
|---|
|  |  |  | // 维护 开放的工位 | 
|---|
|  |  |  | ConcurrentHashMap<String, BedBO> mapOpeningBed = new ConcurrentHashMap<>(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 装机工位..不进入优先队列 | 
|---|
|  |  |  | Map<Integer, PriorityBlockingQueue<BedQueueBO>> mapCheckTypePriorityQueue = new HashMap(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 基于类型的, 装机准备的优先队列 | 
|---|
|  |  |  | // Map<Integer, PriorityBlockingQueue<BedQueueBO>> mapReadyPriorityQueue = new HashMap(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 装机工位..不进入分流队列 | 
|---|
|  |  |  | ConcurrentHashMap<Integer, PriorityBlockingQueue<BedBO>> mapCheckTypeRoutingQueue = new ConcurrentHashMap<>(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | Map<Integer, CheckTypeDO> mapCheckTypeVsReadyMax = null; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | 
|---|
|  |  |  | * @return | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public ErrorCode bedOpen(Long roomId, String roomName, String bedNo) { | 
|---|
|  |  |  | BedQueueBO bedQueueBO2 = mapBedVsQueue.get(String.format("%09d%s", roomId, bedNo)); | 
|---|
|  |  |  | if (null != bedQueueBO2) { | 
|---|
|  |  |  | BedBO bedBO2 = mapOpeningBed.get( Utils.formatRoomBed(roomId, bedNo) ); | 
|---|
|  |  |  | if (null != bedBO2) { | 
|---|
|  |  |  | log.error("bedOpen mapBedVsQueue has existed. " + roomId + " " + bedNo); | 
|---|
|  |  |  | return QUEUE_BED_EXIST; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | return ROOM_INVALID_STATUS; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | List<Byte> queueStatusList = new ArrayList<>(); | 
|---|
|  |  |  | queueStatusList.add(QueueStatusEnum.READY.getStatus()); | 
|---|
|  |  |  | List<QueueDO> queueDOList = queueMapper.getDoctorQueueByStatus(roomId, bedNo, queueStatusList); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | RoomDO roomDO = roomMapper.getRoom(roomId, bedNo); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 新增..工位队列 | 
|---|
|  |  |  | BedQueueBO bedQueueBO = new BedQueueBO(); | 
|---|
|  |  |  | bedQueueBO.setRoomId(roomId); | 
|---|
|  |  |  | bedQueueBO.setRoomName(roomName); | 
|---|
|  |  |  | bedQueueBO.setBedNo(bedNo); | 
|---|
|  |  |  | bedQueueBO.setMaxQueueNum(getBedReadyMax(roomId, bedNo)); | 
|---|
|  |  |  | bedQueueBO.setQueueNum(new AtomicInteger(queueDOList.size())); | 
|---|
|  |  |  | bedQueueBO.setStatus(BedStatusEnum.OPENING.getStatus()); | 
|---|
|  |  |  | bedQueueBO.setCheckTypes( roomDO.getCheckTypes() ); | 
|---|
|  |  |  | bedQueueBO.setOpType( roomDO.getOpType() ); | 
|---|
|  |  |  | mapBedVsQueue.put(String.format("%09d%s", roomId, bedNo), bedQueueBO); | 
|---|
|  |  |  | BedBO bedBO = new BedBO(); | 
|---|
|  |  |  | bedBO.setRoomId(roomId); | 
|---|
|  |  |  | bedBO.setRoomName(roomName); | 
|---|
|  |  |  | bedBO.setBedNo(bedNo); | 
|---|
|  |  |  | bedBO.setStatus(BedStatusEnum.OPENING.getStatus()); | 
|---|
|  |  |  | bedBO.setCheckTypes( roomDO.getCheckTypes() ); | 
|---|
|  |  |  | bedBO.setOpType( roomDO.getOpType() ); | 
|---|
|  |  |  | mapOpeningBed.put( Utils.formatRoomBed(roomId, bedNo), bedBO); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return GlobalErrorCodeConstants.SUCCESS; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public ErrorCode bedClose(Long roomId, String bedNo) { | 
|---|
|  |  |  | BedQueueBO bedQueueBO = mapBedVsQueue.get(String.format("%09d%s", roomId, bedNo)); | 
|---|
|  |  |  | if (null == bedQueueBO) { | 
|---|
|  |  |  | BedBO bedBO = mapOpeningBed.get( Utils.formatRoomBed(roomId, bedNo) ); | 
|---|
|  |  |  | if (null == bedBO) { | 
|---|
|  |  |  | log.error("bedClose mapBedVsQueue DONOT existed. " + roomId + " " + bedNo); | 
|---|
|  |  |  | return QUEUE_BED_NOT_EXIST; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 除了判断 准备候诊中 的人数,还需要 判断 过号的人数 | 
|---|
|  |  |  | //if (bedQueueBO.getQueueNum().get() >0) | 
|---|
|  |  |  | //    return QUEUE_HAVE_PATIENT; | 
|---|
|  |  |  | List<Byte> queueStatusList = new ArrayList<>(); | 
|---|
|  |  |  | // 检查是否还有患者需要处理 | 
|---|
|  |  |  | List<Integer> queueStatusList = new ArrayList<>(); | 
|---|
|  |  |  | queueStatusList.add(QueueStatusEnum.READY.getStatus()); | 
|---|
|  |  |  | queueStatusList.add(QueueStatusEnum.ONSTAGE.getStatus()); | 
|---|
|  |  |  | queueStatusList.add(QueueStatusEnum.INSTALLING.getStatus()); | 
|---|
|  |  |  | queueStatusList.add(QueueStatusEnum.PASSED.getStatus()); | 
|---|
|  |  |  | queueStatusList.add(QueueStatusEnum.PASSED_INSTALL.getStatus()); | 
|---|
|  |  |  | queueStatusList.add(QueueStatusEnum.RECALLED.getStatus()); | 
|---|
|  |  |  | List<QueueDO>  queueDOList = queueMapper.getDoctorQueueByStatus(roomId, bedNo, queueStatusList); | 
|---|
|  |  |  | queueStatusList.add(QueueStatusEnum.RECALLED_INSTALL.getStatus()); | 
|---|
|  |  |  | queueStatusList.add(QueueStatusEnum.AFFINITY.getStatus()); | 
|---|
|  |  |  | queueStatusList.add(QueueStatusEnum.AFFINITY_INSTALL.getStatus()); | 
|---|
|  |  |  | List<QueueDO>  queueDOList = queueMapper.getBedQueueByStatus(roomId, bedNo, queueStatusList); | 
|---|
|  |  |  | if (queueDOList.size() > 0) | 
|---|
|  |  |  | return QUEUE_HAVE_PATIENT; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | List statusList = new ArrayList<BedStatusEnum>(); | 
|---|
|  |  |  | statusList.add(BedStatusEnum.OPENING); | 
|---|
|  |  |  | statusList.add(BedStatusEnum.DOCTOR_ON); | 
|---|
|  |  |  | statusList.add(BedStatusEnum.PAUSE); | 
|---|
|  |  |  | statusList.add(BedStatusEnum.DOCTOR_PAUSE); | 
|---|
|  |  |  | Integer updateNum = roomMapper.setBedOpeningClosed(roomId, bedNo, | 
|---|
|  |  |  | BedStatusEnum.CLOSED, statusList); | 
|---|
|  |  |  | if ( null==updateNum || 0 == updateNum ) { | 
|---|
|  |  |  | 
|---|
|  |  |  | return ROOM_INVALID_STATUS; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | mapBedVsQueue.remove(String.format("%09d%s", roomId, bedNo)); | 
|---|
|  |  |  | removePriorityQueue(bedQueueBO); | 
|---|
|  |  |  | mapOpeningBed.remove( Utils.formatRoomBed(roomId, bedNo) ); | 
|---|
|  |  |  | removeBedFromRoutingQueue(bedBO); | 
|---|
|  |  |  | return GlobalErrorCodeConstants.SUCCESS; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public ErrorCode bedDoctorPause(Long roomId, String bedNo, Long docId, String docName) { | 
|---|
|  |  |  | BedQueueBO bedQueueBO = mapBedVsQueue.get(String.format("%09d%s", roomId, bedNo)); | 
|---|
|  |  |  | if (null == bedQueueBO) { | 
|---|
|  |  |  | BedBO bedBO = mapOpeningBed.get( Utils.formatRoomBed(roomId, bedNo) ); | 
|---|
|  |  |  | if (null == bedBO) { | 
|---|
|  |  |  | log.error("bedDoctorPause mapBedVsQueue DONOT existed. " + roomId + " " + bedNo); | 
|---|
|  |  |  | return QUEUE_BED_NOT_EXIST; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | // DB update | 
|---|
|  |  |  | List statusList = new ArrayList<BedStatusEnum>(); | 
|---|
|  |  |  | statusList.add(BedStatusEnum.DOCTOR_ON); | 
|---|
|  |  |  | Integer updateNum = roomMapper.setBedDoctorStatus(roomId, bedNo, docId, BedStatusEnum.PAUSE, statusList); | 
|---|
|  |  |  | Integer updateNum = roomMapper.setBedDoctorStatus(roomId, bedNo, docId, BedStatusEnum.DOCTOR_PAUSE, statusList); | 
|---|
|  |  |  | if ( null==updateNum || 0 == updateNum ) { | 
|---|
|  |  |  | log.error("bedDoctorPause DB invalid status. " + roomId + " " + bedNo); | 
|---|
|  |  |  | return ROOM_INVALID_STATUS; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | bedQueueBO.setStatus(BedStatusEnum.PAUSE.getStatus()); | 
|---|
|  |  |  | removePriorityQueue(bedQueueBO); | 
|---|
|  |  |  | bedBO.setStatus(BedStatusEnum.DOCTOR_PAUSE.getStatus()); | 
|---|
|  |  |  | removeBedFromRoutingQueue(bedBO); | 
|---|
|  |  |  | return GlobalErrorCodeConstants.SUCCESS; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public ErrorCode bedDoctorResume(Long roomId, String bedNo, Long docId, String docName) { | 
|---|
|  |  |  | BedQueueBO bedQueueBO = mapBedVsQueue.get(String.format("%09d%s", roomId, bedNo)); | 
|---|
|  |  |  | if (null == bedQueueBO || !bedQueueBO.getStatus().equals(BedStatusEnum.PAUSE.getStatus())) { | 
|---|
|  |  |  | BedBO bedBO = mapOpeningBed.get( Utils.formatRoomBed(roomId, bedNo) ); | 
|---|
|  |  |  | if (null == bedBO || !bedBO.getStatus().equals(BedStatusEnum.DOCTOR_PAUSE.getStatus())) { | 
|---|
|  |  |  | log.error("bedDoctorResume mapBedVsQueue DONOT existed OR NOT Paused. " + roomId + " " + bedNo); | 
|---|
|  |  |  | return QUEUE_BED_NOT_EXIST; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // DB update | 
|---|
|  |  |  | List statusList = new ArrayList<BedStatusEnum>(); | 
|---|
|  |  |  | statusList.add(BedStatusEnum.PAUSE); | 
|---|
|  |  |  | statusList.add(BedStatusEnum.DOCTOR_PAUSE); | 
|---|
|  |  |  | Integer updateNum = roomMapper.setBedDoctorStatus(roomId, bedNo, docId, | 
|---|
|  |  |  | BedStatusEnum.DOCTOR_ON, statusList); | 
|---|
|  |  |  | if ( null==updateNum || 0 == updateNum ) { | 
|---|
|  |  |  | 
|---|
|  |  |  | return ROOM_INVALID_STATUS; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | bedQueueBO.setStatus(BedStatusEnum.DOCTOR_ON.getStatus()); | 
|---|
|  |  |  | addPriorityQueue(bedQueueBO); | 
|---|
|  |  |  | bedBO.setStatus(BedStatusEnum.DOCTOR_ON.getStatus()); | 
|---|
|  |  |  | addBedToRoutingQueue(bedBO); | 
|---|
|  |  |  | return GlobalErrorCodeConstants.SUCCESS; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public ErrorCode bedDoctorOn(Long roomId, String bedNo, Long docId, String docName) { | 
|---|
|  |  |  | BedQueueBO bedQueueBO = mapBedVsQueue.get(String.format("%09d%s", roomId, bedNo)); | 
|---|
|  |  |  | if (null == bedQueueBO) { | 
|---|
|  |  |  | BedBO bedBO = mapOpeningBed.get( Utils.formatRoomBed(roomId, bedNo) ); | 
|---|
|  |  |  | if (null == bedBO) { | 
|---|
|  |  |  | log.error("bedDoctorOn mapBedVsQueue DONOT existed. " + roomId + " " + bedNo); | 
|---|
|  |  |  | return QUEUE_BED_NOT_EXIST; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | return ROOM_INVALID_STATUS; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | bedQueueBO.setStatus(BedStatusEnum.DOCTOR_ON.getStatus()); | 
|---|
|  |  |  | addPriorityQueue(bedQueueBO); | 
|---|
|  |  |  | bedBO.setStatus(BedStatusEnum.DOCTOR_ON.getStatus()); | 
|---|
|  |  |  | addBedToRoutingQueue(bedBO); | 
|---|
|  |  |  | return GlobalErrorCodeConstants.SUCCESS; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public ErrorCode bedDoctorOff(Long roomId, String bedNo, Long docId, String docName) { | 
|---|
|  |  |  | BedQueueBO bedQueueBO = mapBedVsQueue.get(String.format("%09d%s", roomId, bedNo)); | 
|---|
|  |  |  | if (null == bedQueueBO) { | 
|---|
|  |  |  | BedBO bedBO = mapOpeningBed.get( Utils.formatRoomBed( roomId, bedNo ) ); | 
|---|
|  |  |  | if (null == bedBO) { | 
|---|
|  |  |  | log.error("bedDoctorOff mapBedVsQueue DONOT existed. " + roomId + " " + bedNo); | 
|---|
|  |  |  | return QUEUE_BED_NOT_EXIST; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | // DB update | 
|---|
|  |  |  | List statusList = new ArrayList<BedStatusEnum>(); | 
|---|
|  |  |  | statusList.add(BedStatusEnum.DOCTOR_ON); | 
|---|
|  |  |  | statusList.add(BedStatusEnum.PAUSE); | 
|---|
|  |  |  | statusList.add(BedStatusEnum.DOCTOR_PAUSE); | 
|---|
|  |  |  | Integer updateNum = roomMapper.setBedDoctorOff(roomId, bedNo, docId, BedStatusEnum.OPENING, statusList); | 
|---|
|  |  |  | if ( null==updateNum || 0 == updateNum ) { | 
|---|
|  |  |  | log.error("bedDoctorOff DB invalid status. " + roomId + " " + bedNo); | 
|---|
|  |  |  | return ROOM_INVALID_STATUS; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | removePriorityQueue(bedQueueBO); | 
|---|
|  |  |  | removeBedFromRoutingQueue(bedBO); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | bedQueueBO.setStatus(BedStatusEnum.OPENING.getStatus()); | 
|---|
|  |  |  | bedBO.setStatus(BedStatusEnum.OPENING.getStatus()); | 
|---|
|  |  |  | return GlobalErrorCodeConstants.SUCCESS; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /* 2024.11.11 | 
|---|
|  |  |  | // 常规 或者 领用 时 医生 取下一位患者 | 
|---|
|  |  |  | public ErrorCode nextPatient(Long roomId, String bedNo) { | 
|---|
|  |  |  | // 从 DB 把 序号最小的 就诊准备中的人 设置为就诊中 | 
|---|
|  |  |  | Integer updateNum = queueMapper.updateQueueStatus(roomId, bedNo, | 
|---|
|  |  |  | // 从 DB 把该工位.序号最小的 [候诊中中] 的人 设置为 [就诊中] (或领用中) | 
|---|
|  |  |  | Integer updateNum = queueMapper.updateBedQueueStatus(roomId, bedNo, | 
|---|
|  |  |  | QueueStatusEnum.READY.getStatus(), QueueStatusEnum.ONSTAGE.getStatus()); | 
|---|
|  |  |  | // 该工位 没有 就诊准备中 人员 | 
|---|
|  |  |  | // 该工位 没有 [候诊中中] 人员 | 
|---|
|  |  |  | if (null == updateNum || 0 == updateNum) { | 
|---|
|  |  |  | return QUEUE_NOT_READY_PATIENT; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 优先队列中 该工位 就诊准备中人的数量 减一 | 
|---|
|  |  |  | BedQueueBO bo = mapBedVsQueue.get(String.format("%09d%s", roomId, bedNo)); | 
|---|
|  |  |  | // 分流队列中 该工位 候诊中中人的数量 减一 | 
|---|
|  |  |  | BedQueueBO bo = mapBedVsQueue.get( Utils.formatRoomBed( roomId, bedNo ) ); | 
|---|
|  |  |  | bo.queueNum.getAndDecrement(); // 可能已经【并发的】在 hurry-up 中改变了值 | 
|---|
|  |  |  |  | 
|---|
|  |  |  | removePriorityQueue(bo); | 
|---|
|  |  |  | refreshRoutingQueue(bo); | 
|---|
|  |  |  | return GlobalErrorCodeConstants.SUCCESS; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | */ | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 常规 或者 领用 时 医生 取下一位患者 | 
|---|
|  |  |  | public ErrorCode nextPatient(Long roomId, String bedNo) { | 
|---|
|  |  |  | QueueDO firstReadyQueueItem = queueMapper.getBedFirstQueueItemByStatus(roomId, bedNo, QueueStatusEnum.READY.getStatus()); | 
|---|
|  |  |  | // 该工位 没有 [候诊中中] 人员 | 
|---|
|  |  |  | if (null == firstReadyQueueItem) { | 
|---|
|  |  |  | return QUEUE_NOT_READY_PATIENT; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // QueueStatusEnum.READY  =>  QueueStatusEnum.ONSTAGE | 
|---|
|  |  |  | firstReadyQueueItem.setStatus(QueueStatusEnum.ONSTAGE.getStatus()); | 
|---|
|  |  |  | queueMapper.updateById( firstReadyQueueItem ); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 分流队列中 该工位 【候诊中】患者数量 减一 | 
|---|
|  |  |  | BedBO bo = mapOpeningBed.get( Utils.formatRoomBed( roomId, bedNo ) ); | 
|---|
|  |  |  | bo.readyPatNum--; // 可能已经在 hurry-up 中改变了值 | 
|---|
|  |  |  |  | 
|---|
|  |  |  | refreshBedPriorityLevelInRoutingQueue(bo); | 
|---|
|  |  |  | return GlobalErrorCodeConstants.SUCCESS; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 装机时 医生 取下一位患者 | 
|---|
|  |  |  | public ErrorCode nextInstallPatient(Long roomId, String bedNo) { | 
|---|
|  |  |  | // 从 DB 把 序号最小的 [已领用] [已召回-安装] 的人 设置为 [安装中] | 
|---|
|  |  |  | QueueDO firstItem = queueMapper.getRoomFirstQueueItemByStatus(roomId, List.of(QueueStatusEnum.RECEIVED.getStatus(), QueueStatusEnum.RECALLED_INSTALL.getStatus())); | 
|---|
|  |  |  | // 该工位 没有 [已领用] | [已召回-安装] 人员 | 
|---|
|  |  |  | if (null == firstItem) { | 
|---|
|  |  |  | return QUEUE_NOT_READY_PATIENT; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | firstItem.setStatus( QueueStatusEnum.INSTALLING.getStatus() ); | 
|---|
|  |  |  | firstItem.setBedNo( bedNo );  // 转到 安装工位 | 
|---|
|  |  |  | queueMapper.updateById( firstItem ); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 检查项目.亲和性 处理逻辑:  变为【安装中】后 拉 其他检查项来亲和 | 
|---|
|  |  |  | CheckTypeDO checkTypeDO = getCheckTypeItem( firstItem.getBookCheckType() ); | 
|---|
|  |  |  | if (checkTypeDO.getAffinityCheckTypes().length > 0) { | 
|---|
|  |  |  | procAffinityWhenSomeOneEntryInstalling(firstItem.getPatId(), roomId, bedNo, checkTypeDO.getAffinityCheckTypes() ); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 装机工位 不参与分流 | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return GlobalErrorCodeConstants.SUCCESS; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 1. 每天开诊前 从DB同步工位的患者队列数据到 工位优先队列 | 
|---|
|  |  |  | * 1. 每天开诊前 从DB同步工位的患者队列数据到 工位分流队列 | 
|---|
|  |  |  | * 2. 服务运维重启时 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public ErrorCode bedReload() { | 
|---|
|  |  |  | mapCheckTypePriorityQueue.clear(); | 
|---|
|  |  |  | mapBedVsQueue.clear(); | 
|---|
|  |  |  | mapCheckTypeRoutingQueue.clear(); | 
|---|
|  |  |  | mapOpeningBed.clear(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 清除非当天的排队人员 | 
|---|
|  |  |  | queueMapper.clearQueue(); | 
|---|
|  |  |  | 
|---|
|  |  |  | List<BedStatusEnum> bedStatusEnumList = new ArrayList<BedStatusEnum>(); | 
|---|
|  |  |  | bedStatusEnumList.add(BedStatusEnum.OPENING); | 
|---|
|  |  |  | bedStatusEnumList.add(BedStatusEnum.DOCTOR_ON); | 
|---|
|  |  |  | bedStatusEnumList.add(BedStatusEnum.PAUSE); | 
|---|
|  |  |  | bedStatusEnumList.add(BedStatusEnum.DOCTOR_PAUSE); | 
|---|
|  |  |  | List<RoomDO> roomDOList = roomMapper.simpleRoomList(bedStatusEnumList); | 
|---|
|  |  |  | List<BedQueueBO> bedQueueBOList = roomDOList.stream().map(item -> BeanUtils.toBean(item, BedQueueBO.class)).toList(); | 
|---|
|  |  |  | List<BedBO> bedBOList = roomDOList.stream().map(item -> BeanUtils.toBean(item, BedBO.class)).toList(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 从DB 获取 队列中 就诊准备中人员统计 列表 | 
|---|
|  |  |  | List<Byte> queueStatusList = new ArrayList<>(); | 
|---|
|  |  |  | queueStatusList.add(QueueStatusEnum.READY.getStatus()); | 
|---|
|  |  |  | List<QueueStatisticDO> queueStatisticDOList = queueMapper.queueStatistic(queueStatusList); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | bedQueueBOList.forEach(item -> { | 
|---|
|  |  |  | item.maxQueueNum = getBedReadyMax(item.roomId, item.bedNo); | 
|---|
|  |  |  | Optional<QueueStatisticDO> queueStatisticDOOptional = queueStatisticDOList.stream().filter(it->it.getRoomId()==item.roomId && it.getBedNo().equals(item.getBedNo())).findFirst(); | 
|---|
|  |  |  | int queueNum = queueStatisticDOOptional.isPresent() ? queueStatisticDOOptional.get().getTotalInStatus() : 0; | 
|---|
|  |  |  | if ( item.maxQueueNum < queueNum ) | 
|---|
|  |  |  | throw new RuntimeException("init: exceed max queue number!"); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | item.queueNum.set( queueNum ); | 
|---|
|  |  |  | mapBedVsQueue.put(String.format("%09d%s", item.roomId, item.bedNo), item); | 
|---|
|  |  |  | bedBOList.forEach(item -> { | 
|---|
|  |  |  | mapOpeningBed.put( Utils.formatRoomBed(item.roomId, item.bedNo), item); | 
|---|
|  |  |  | if (BedStatusEnum.DOCTOR_ON.getStatus() == item.getStatus()) { | 
|---|
|  |  |  | addPriorityQueue(item); | 
|---|
|  |  |  | addBedToRoutingQueue(item); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public void resetRoom(Boolean needCloseBed) { | 
|---|
|  |  |  | // 踢出在座的医生 | 
|---|
|  |  |  | List<BedStatusEnum> bedStatusEnumList = new ArrayList<BedStatusEnum>(); | 
|---|
|  |  |  | bedStatusEnumList.add(BedStatusEnum.DOCTOR_ON); | 
|---|
|  |  |  | bedStatusEnumList.add(BedStatusEnum.PAUSE); | 
|---|
|  |  |  | List<RoomDO> roomDOList = roomMapper.simpleRoomList(bedStatusEnumList); | 
|---|
|  |  |  | List<Long> userIdList = roomDOList.stream().map(roomDO -> roomDO.getDocId()).toList(); | 
|---|
|  |  |  | oAuth2TokenApi.tick(userIdList); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 关闭所有工位 | 
|---|
|  |  |  | if (needCloseBed) { | 
|---|
|  |  |  | // 踢出在座的医生 | 
|---|
|  |  |  | List<BedStatusEnum> bedStatusEnumList = new ArrayList<BedStatusEnum>(); | 
|---|
|  |  |  | bedStatusEnumList.add(BedStatusEnum.DOCTOR_ON); | 
|---|
|  |  |  | bedStatusEnumList.add(BedStatusEnum.DOCTOR_PAUSE); | 
|---|
|  |  |  | List<RoomDO> roomDOList = roomMapper.simpleRoomList(bedStatusEnumList); | 
|---|
|  |  |  | List<Long> userIdList = roomDOList.stream().map(roomDO -> roomDO.getDocId()).toList(); | 
|---|
|  |  |  | oAuth2TokenApi.tick(userIdList); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 所有工位状态 => 关闭 | 
|---|
|  |  |  | Integer ret = roomMapper.resetRoom(BedStatusEnum.CLOSED.getStatus()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | if (0 == openingFlag.get()) | 
|---|
|  |  |  | return; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | mapBedVsQueue.values().forEach(bedQueueBO -> hurryupOneBed(bedQueueBO.roomId, bedQueueBO.bedNo)); | 
|---|
|  |  |  | mapOpeningBed.values().forEach(bedQueueBO -> hurryupOneBed(bedQueueBO.roomId, bedQueueBO.bedNo)); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | 
|---|
|  |  |  | if (0 == openingFlag.get()) | 
|---|
|  |  |  | return; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | BedQueueBO bedQueueBO = getBedQueueBO(roomId, bedNo); | 
|---|
|  |  |  | BedBO bedBO = getBedQueueBO(roomId, bedNo); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 处理 过号-回来 的人 | 
|---|
|  |  |  | while (bedQueueBO.queueNum.get() < bedQueueBO.maxQueueNum) { | 
|---|
|  |  |  | while (bedBO.getReadyPatNum() < bedBO.getMaxReadyPatNum()) { | 
|---|
|  |  |  | // 查看 当前工位 是否有过号-回来的患者 | 
|---|
|  |  |  | Integer updateNum = queueMapper.queueRecalledPatient( | 
|---|
|  |  |  | bedQueueBO.getRoomId(), | 
|---|
|  |  |  | bedQueueBO.getRoomName(), | 
|---|
|  |  |  | bedQueueBO.getBedNo(), | 
|---|
|  |  |  | curSeqNum.get() + 1, | 
|---|
|  |  |  | QueueStatusEnum.RECALLED.getStatus(), | 
|---|
|  |  |  | bedBO.getRoomId(), | 
|---|
|  |  |  | bedBO.getRoomName(), | 
|---|
|  |  |  | bedBO.getBedNo(), | 
|---|
|  |  |  | curSeqNum.get() + 1,  // 过号回来,排到当前的下一位 | 
|---|
|  |  |  | QueueStatusEnum.RECALLED.getStatus(), // 不需要处理 [安装召回], 原因: 安装工位 不进入分流队列 | 
|---|
|  |  |  | QueueStatusEnum.READY.getStatus()); | 
|---|
|  |  |  | if (null == updateNum || 0 == updateNum) | 
|---|
|  |  |  | break; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // [过号] 患者 => [候诊中] ,处理 检查项目.亲和性 逻辑 | 
|---|
|  |  |  | QueueDO recalledQueueItem = queueMapper.getQueueItemBySeqNum(curSeqNum.get() + 1); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | CheckTypeDO checkTypeDO = getCheckTypeItem( recalledQueueItem.getBookCheckType() ); | 
|---|
|  |  |  | if ( checkTypeDO.getAffinityCheckTypes().length > 0) { | 
|---|
|  |  |  | procAffinityWhenRecalled(recalledQueueItem, checkTypeDO.getAffinityCheckTypes()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | curSeqNum.getAndIncrement(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 可能已经【并发的】在 nextPatient 中改变了值 | 
|---|
|  |  |  | bedQueueBO.queueNum.incrementAndGet(); | 
|---|
|  |  |  | // 可能已经在 nextPatient 中改变了值 | 
|---|
|  |  |  | bedBO.readyPatNum++; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 可能已经【并发的】在 nextPatient 中改变了优先队列顺序 | 
|---|
|  |  |  | refreshPriorityQueue(bedQueueBO); | 
|---|
|  |  |  | // 可能已经【并发的】在 nextPatient 中改变了分流队列顺序 | 
|---|
|  |  |  | refreshBedPriorityLevelInRoutingQueue(bedBO); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 处理 排队中 患者 | 
|---|
|  |  |  | Integer[] checkTypes = bedQueueBO.getCheckTypes(); | 
|---|
|  |  |  | Arrays.stream(checkTypes).forEach(checkType -> hurryupOneCheckType(checkType)); | 
|---|
|  |  |  | hurryupOneBedCheckTypeList(bedBO); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 一个工位可以支持多个检查项目,所以需要保证多个检查项目的就诊时间先后顺序 | 
|---|
|  |  |  | * 按照一个检查项目来处理,就不能保证同一个工位的多个检查项目的就诊时间的先后顺序 | 
|---|
|  |  |  | * 所以实现逻辑是 让支持该检查项目的所有工位去抢占 【排队中】【工位支持的检查项目】的患者 | 
|---|
|  |  |  | * 实际抢占的未必是 参数中的检查项目的患者,可能是其他检查项目的患者 | 
|---|
|  |  |  | * @param checkType | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public void hurryupOneCheckType(Integer checkType) { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | PriorityBlockingQueue<BedQueueBO> priorityQueue = mapCheckTypePriorityQueue.get(checkType); | 
|---|
|  |  |  | if (null == priorityQueue) | 
|---|
|  |  |  | PriorityBlockingQueue<BedBO> routingQueue = mapCheckTypeRoutingQueue.get(checkType); | 
|---|
|  |  |  | if (null == routingQueue) | 
|---|
|  |  |  | return; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | while (true) { | 
|---|
|  |  |  | BedQueueBO bedQueueBO = priorityQueue.peek(); | 
|---|
|  |  |  | if (null == bedQueueBO) | 
|---|
|  |  |  | BedBO bedBO = routingQueue.peek(); | 
|---|
|  |  |  | if (null == bedBO) | 
|---|
|  |  |  | return; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | int curQueueNum = bedQueueBO.queueNum.get(); | 
|---|
|  |  |  | if (curQueueNum > bedQueueBO.maxQueueNum) | 
|---|
|  |  |  | int curQueueNum = bedBO.getReadyPatNum(); | 
|---|
|  |  |  | if (curQueueNum > bedBO.getMaxReadyPatNum()) | 
|---|
|  |  |  | throw new RuntimeException("hurryupOneCheckType: exceed max queue number!"); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (curQueueNum == bedQueueBO.maxQueueNum) | 
|---|
|  |  |  | if (curQueueNum == bedBO.getMaxReadyPatNum()) | 
|---|
|  |  |  | return; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 查看 是否有排队中的患者 | 
|---|
|  |  |  | Integer updateNum = queueMapper.preemptPatient( | 
|---|
|  |  |  | bedQueueBO.getRoomId(), | 
|---|
|  |  |  | bedQueueBO.getRoomName(), | 
|---|
|  |  |  | bedQueueBO.getBedNo(), | 
|---|
|  |  |  | Integer updateNum = queueMapper.preemptWaitingPatientWithBedCheckTypes( | 
|---|
|  |  |  | bedBO.getRoomId(), | 
|---|
|  |  |  | bedBO.getRoomName(), | 
|---|
|  |  |  | bedBO.getBedNo(), | 
|---|
|  |  |  | curSeqNum.get() + 1, | 
|---|
|  |  |  | QueueStatusEnum.WAITING.getStatus(), | 
|---|
|  |  |  | QueueStatusEnum.READY.getStatus(), | 
|---|
|  |  |  | checkType); | 
|---|
|  |  |  | bedBO.checkTypes);  //关键点 关键点 关键点 checkType | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 没有抢到排队患者 | 
|---|
|  |  |  | if (null == updateNum || 0 == updateNum) { | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | QueueDO preemptQueueItem = queueMapper.getQueueItemBySeqNum(curSeqNum.get() + 1); | 
|---|
|  |  |  | CheckTypeDO checkTypeDO = getCheckTypeItem( preemptQueueItem.getBookCheckType() ); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 抢到排队患者后,处理 检查项目.亲和性 逻辑 | 
|---|
|  |  |  | if ( checkTypeDO.getAffinityCheckTypes().length > 0) { | 
|---|
|  |  |  | procAffinityWhenPreempt(preemptQueueItem, checkTypeDO.getAffinityCheckTypes()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | curSeqNum.getAndIncrement(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 可能已经【并发的】在 nextPatient 中改变了值 | 
|---|
|  |  |  | bedQueueBO.queueNum.incrementAndGet(); | 
|---|
|  |  |  | // 可能已经在 nextPatient 中改变了值 | 
|---|
|  |  |  | bedBO.readyPatNum++; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 可能已经【并发的】在 nextPatient 中改变了优先队列顺序 | 
|---|
|  |  |  | refreshPriorityQueue(bedQueueBO); | 
|---|
|  |  |  | // 可能已经【并发的】在 nextPatient 中改变了分流队列顺序 | 
|---|
|  |  |  | refreshBedPriorityLevelInRoutingQueue(bedBO); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | mapCheckTypeVsReadyMax = checkTypeDOList.stream().collect(Collectors.toMap(CheckTypeDO::getValue, checkType -> checkType)); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public BedQueueBO getBedQueueBO(Long roomId, String bedNo) { | 
|---|
|  |  |  | return mapBedVsQueue.get(String.format("%09d%s", roomId, bedNo)); | 
|---|
|  |  |  | public CheckTypeDO getCheckTypeItem(Integer checkType) { | 
|---|
|  |  |  | return mapCheckTypeVsReadyMax.get( checkType ); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public BedBO getBedQueueBO(Long roomId, String bedNo) { | 
|---|
|  |  |  | return mapOpeningBed.get( Utils.formatRoomBed(roomId, bedNo) ); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public MonitorInfoVO getMonitorInfo() { | 
|---|
|  |  |  | 
|---|
|  |  |  | public void monitorInfo() { | 
|---|
|  |  |  | MonitorInfoVO monitorInfoVO = new MonitorInfoVO(); | 
|---|
|  |  |  | roomBedStatistic(monitorInfoVO); | 
|---|
|  |  |  | log.info(" opening " + openingFlag.get() + " " + monitorInfoVO.getQueueNum() + " " + monitorInfoVO.getActiveQueueNum() + " " + monitorInfoVO.getCheckTypeBedInfo().toString() ); | 
|---|
|  |  |  | log.info(" opening " + openingFlag.get() + " " + monitorInfoVO.getOpeningBedNum() + " " + monitorInfoVO.getDocBedNum() + " " + monitorInfoVO.getCheckTypeBedInfo().toString() ); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private Integer getBedReadyMax(Long roomId, String bedNo) { | 
|---|
|  |  |  | RoomDO roomDO = roomMapper.getRoom(roomId, bedNo); | 
|---|
|  |  |  | Integer[] checkTypes = roomDO.getCheckTypes(); | 
|---|
|  |  |  | Integer ret = 0; | 
|---|
|  |  |  | for (int i=0; i<checkTypes.length; i++) { | 
|---|
|  |  |  | CheckTypeDO checkTypeDO = mapCheckTypeVsReadyMax.get(checkTypes[i]); | 
|---|
|  |  |  | private void hurryupOneBedCheckTypeList(BedBO bedBO) { | 
|---|
|  |  |  | while (bedBO.getReadyPatNum() < bedBO.getMaxReadyPatNum()) { | 
|---|
|  |  |  | // 抢占 是否有排队中的患者 | 
|---|
|  |  |  | Integer updateNum = queueMapper.preemptWaitingPatientWithBedCheckTypes( | 
|---|
|  |  |  | bedBO.getRoomId(), | 
|---|
|  |  |  | bedBO.getRoomName(), | 
|---|
|  |  |  | bedBO.getBedNo(), | 
|---|
|  |  |  | curSeqNum.get() + 1, | 
|---|
|  |  |  | QueueStatusEnum.WAITING.getStatus(), | 
|---|
|  |  |  | QueueStatusEnum.READY.getStatus(), | 
|---|
|  |  |  | bedBO.checkTypes); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 没有抢到排队患者 | 
|---|
|  |  |  | if (null == updateNum || 0 == updateNum) { | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | QueueDO preemptQueueItem = queueMapper.getQueueItemBySeqNum(curSeqNum.get() + 1); | 
|---|
|  |  |  | CheckTypeDO checkTypeDO = getCheckTypeItem( preemptQueueItem.getBookCheckType() ); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 抢到排队患者后,处理 检查项目.亲和性 逻辑 | 
|---|
|  |  |  | if ( checkTypeDO.getAffinityCheckTypes().length > 0) { | 
|---|
|  |  |  | procAffinityWhenPreempt(preemptQueueItem, checkTypeDO.getAffinityCheckTypes()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | curSeqNum.getAndIncrement(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | bedBO.readyPatNum++; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 改变了分流队列顺序 | 
|---|
|  |  |  | refreshBedPriorityLevelInRoutingQueue(bedBO); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 有检查项分流到诊室后,也就是进入[候诊中]后,拉 [排队中] + 具有亲和性的 检查项 来依附该检查项 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | private void procAffinityWhenPreempt(QueueDO preemptQueueItem, Integer[] affinityCheckTypes) { | 
|---|
|  |  |  | List<QueueDO> affinityItems = queueMapper.getCurPatGivenCheckTypesAndStatus( | 
|---|
|  |  |  | preemptQueueItem.getPatId(), affinityCheckTypes, QueueStatusEnum.WAITING.getStatus()); | 
|---|
|  |  |  | for (int i = 0; i < affinityItems.size(); i++) { | 
|---|
|  |  |  | QueueDO queueItem = affinityItems.get(i); | 
|---|
|  |  |  | queueItem.setStatus(QueueStatusEnum.AFFINITY.getStatus()); //改变 排队状态 | 
|---|
|  |  |  | queueItem.setAffinityItem( 1 ); | 
|---|
|  |  |  | queueItem.setRoomId(preemptQueueItem.getRoomId()); | 
|---|
|  |  |  | queueItem.setRoomName(preemptQueueItem.getRoomName()); | 
|---|
|  |  |  | queueItem.setBedNo(preemptQueueItem.getBedNo()); | 
|---|
|  |  |  | queueItem.setSeqNum( preemptQueueItem.getSeqNum());  // 使用同一个 内部序号 | 
|---|
|  |  |  | queueMapper.updateById(queueItem); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * [过号] 患者 => [候诊中] ,处理 检查项目.亲和性 逻辑 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | private void procAffinityWhenRecalled(QueueDO recalledQueueItem, Integer[] affinityCheckTypes) { | 
|---|
|  |  |  | List<QueueDO> affinityItems = queueMapper.getCurPatGivenCheckTypesAndStatus( | 
|---|
|  |  |  | recalledQueueItem.getPatId(), affinityCheckTypes, QueueStatusEnum.AFFINITY.getStatus()); | 
|---|
|  |  |  | for (int i = 0; i < affinityItems.size(); i++) { | 
|---|
|  |  |  | QueueDO queueItem = affinityItems.get(i); | 
|---|
|  |  |  | queueItem.setSeqNum( recalledQueueItem.getSeqNum() );  // [过号] 患者 => [候诊中],原内部序号更新 | 
|---|
|  |  |  | queueMapper.updateById(queueItem); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private Integer getBedReadyMax(BedBO bedBO) { | 
|---|
|  |  |  | Integer[] checkTypes = bedBO.getCheckTypes(); | 
|---|
|  |  |  | int ret = 0; | 
|---|
|  |  |  | for (Integer checkType : checkTypes) { | 
|---|
|  |  |  | CheckTypeDO checkTypeDO = mapCheckTypeVsReadyMax.get(checkType); | 
|---|
|  |  |  | Integer tmp = checkTypeDO.getReadyNum(); | 
|---|
|  |  |  | ret = ret < tmp ? tmp : ret;  //取大值 | 
|---|
|  |  |  | ret = ret < tmp ? tmp : ret;  //【候诊中】数量 取大值 | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return ret; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private void addPriorityQueue(BedQueueBO bedQueueBO) { | 
|---|
|  |  |  | Integer[] checkTypes = bedQueueBO.getCheckTypes(); | 
|---|
|  |  |  | private void addBedToRoutingQueue(BedBO bedBO) { | 
|---|
|  |  |  | // 装机工位 不进 分流队列,不接收 【排队中】患者分流 | 
|---|
|  |  |  | if (Objects.equals(BedOpTypeEnum.OP_INSTALL.getOpType(), bedBO.getOpType())) | 
|---|
|  |  |  | return; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 同步DB, 设置工位的分流控制信息 | 
|---|
|  |  |  | List<Integer> queueStatusList = new ArrayList<>(); | 
|---|
|  |  |  | queueStatusList.add(QueueStatusEnum.READY.getStatus()); //[候诊中] | 
|---|
|  |  |  | List<QueueDO> queueDOList = queueMapper.getBedQueueByStatus(bedBO.getRoomId(), bedBO.getBedNo(), queueStatusList); | 
|---|
|  |  |  | bedBO.setMaxReadyPatNum( getBedReadyMax(bedBO) ); | 
|---|
|  |  |  | bedBO.setReadyPatNum( queueDOList.size() ); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | Integer[] checkTypes = bedBO.getCheckTypes(); | 
|---|
|  |  |  | Arrays.stream(checkTypes).forEach(checkType -> { | 
|---|
|  |  |  | CheckTypeDO checkTypeDO = mapCheckTypeVsReadyMax.get(checkType); | 
|---|
|  |  |  | // 装机工位 不进 优先队列,不从排队中接收人 | 
|---|
|  |  |  | if (checkTypeDO.getNeedDevReady() == 1 && bedQueueBO.getOpType() == 2) { | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | PriorityBlockingQueue<BedQueueBO> priorityQueue = mapCheckTypePriorityQueue.get(checkType); | 
|---|
|  |  |  | if (null == priorityQueue) { | 
|---|
|  |  |  | priorityQueue = new PriorityBlockingQueue<BedQueueBO>(); | 
|---|
|  |  |  | mapCheckTypePriorityQueue.put( checkType, priorityQueue ); | 
|---|
|  |  |  | PriorityBlockingQueue<BedBO> routingQueue = mapCheckTypeRoutingQueue.get(checkType); | 
|---|
|  |  |  | if (null == routingQueue) { | 
|---|
|  |  |  | routingQueue = new PriorityBlockingQueue<BedBO>(); | 
|---|
|  |  |  | mapCheckTypeRoutingQueue.put( checkType, routingQueue ); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (!priorityQueue.contains(bedQueueBO)) { | 
|---|
|  |  |  | priorityQueue.offer(bedQueueBO); | 
|---|
|  |  |  | if (!routingQueue.contains(bedBO)) { | 
|---|
|  |  |  | routingQueue.offer(bedBO); | 
|---|
|  |  |  | } else { | 
|---|
|  |  |  | log.error("bedDoctorResume priorityQueue still exist!. " + bedQueueBO.getRoomId() + " " + bedQueueBO.getBedNo()); | 
|---|
|  |  |  | log.error("bedDoctorResume routingQueue still exist!. " + bedBO.getRoomId() + " " + bedBO.getBedNo()); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private void removePriorityQueue(BedQueueBO bedQueueBO) { | 
|---|
|  |  |  | Integer[] checkTypes = bedQueueBO.getCheckTypes(); | 
|---|
|  |  |  | private void removeBedFromRoutingQueue(BedBO bedBO) { | 
|---|
|  |  |  | // 装机工位 不进 分流队列,不接收 【排队中】患者分流 | 
|---|
|  |  |  | if (Objects.equals(BedOpTypeEnum.OP_INSTALL.getOpType(), bedBO.getOpType())) | 
|---|
|  |  |  | return; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | Integer[] checkTypes = bedBO.getCheckTypes(); | 
|---|
|  |  |  | Arrays.stream(checkTypes).forEach(checkType -> { | 
|---|
|  |  |  | PriorityBlockingQueue<BedQueueBO> priorityBlockingQueue = mapCheckTypePriorityQueue.get(checkType); | 
|---|
|  |  |  | if (null != priorityBlockingQueue) | 
|---|
|  |  |  | priorityBlockingQueue.remove(bedQueueBO); | 
|---|
|  |  |  | PriorityBlockingQueue<BedBO> routingQueue = mapCheckTypeRoutingQueue.get(checkType); | 
|---|
|  |  |  | if (null != routingQueue) | 
|---|
|  |  |  | routingQueue.remove(bedBO); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private void refreshPriorityQueue(BedQueueBO bedQueueBO) { | 
|---|
|  |  |  | Integer[] checkTypes = bedQueueBO.getCheckTypes(); | 
|---|
|  |  |  | private void refreshBedPriorityLevelInRoutingQueue(BedBO bedBO) { | 
|---|
|  |  |  | // 装机工位 不进 分流队列,不接收 【排队中】患者分流 | 
|---|
|  |  |  | if (Objects.equals(BedOpTypeEnum.OP_INSTALL.getOpType(), bedBO.getOpType())) | 
|---|
|  |  |  | return; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | Integer[] checkTypes = bedBO.getCheckTypes(); | 
|---|
|  |  |  | Arrays.stream(checkTypes).forEach(checkType -> { | 
|---|
|  |  |  | PriorityBlockingQueue<BedQueueBO> priorityQueue = mapCheckTypePriorityQueue.get(checkType); | 
|---|
|  |  |  | priorityQueue.remove(bedQueueBO); | 
|---|
|  |  |  | priorityQueue.offer(bedQueueBO); | 
|---|
|  |  |  | PriorityBlockingQueue<BedBO> routingQueue = mapCheckTypeRoutingQueue.get(checkType); | 
|---|
|  |  |  | if (null != routingQueue) { | 
|---|
|  |  |  | routingQueue.remove(bedBO); | 
|---|
|  |  |  | routingQueue.offer(bedBO); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 检查项进入到 [安装中] 时,拉 [已领用] + 具有亲和性的 检查项 来依附该检查项 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | private void procAffinityWhenSomeOneEntryInstalling(String patId, Long roomId_operator, String bedNo_operator, Integer[] affinityCheckTypes) { | 
|---|
|  |  |  | List<QueueDO> affinityItems = queueMapper.getCurPatGivenCheckTypesAndStatus( | 
|---|
|  |  |  | patId, affinityCheckTypes, QueueStatusEnum.RECEIVED.getStatus()); | 
|---|
|  |  |  | for (int i = 0; i < affinityItems.size(); i++) { | 
|---|
|  |  |  | QueueDO queueItem = affinityItems.get(i); | 
|---|
|  |  |  | queueItem.setStatus(QueueStatusEnum.AFFINITY_INSTALL.getStatus()); //改变 排队状态 | 
|---|
|  |  |  | queueItem.setAffinityItem(1); | 
|---|
|  |  |  | queueItem.setRoomId(roomId_operator); | 
|---|
|  |  |  | queueItem.setBedNo(bedNo_operator);  // 跟随到 装机 工位 | 
|---|
|  |  |  | queueMapper.updateById(queueItem); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | private void roomBedStatistic(MonitorInfoVO monitorInfoVO) { | 
|---|
|  |  |  | Map<Integer, Integer>  mapOpeningCheckBedStatInfo = new HashMap<>(); | 
|---|
|  |  |  | Map<Integer, Integer>  mapOpeningInstallBedStatInfo = new HashMap<>(); | 
|---|
|  |  |  | Map<Integer, Integer>  mapOperatingCheckBedStatInfo = new HashMap<>(); | 
|---|
|  |  |  | Map<Integer, Integer>  mapOperatingInstallBedStatInfo = new HashMap<>(); | 
|---|
|  |  |  | Map<Integer, Integer>  mapDocCheckBedStatInfo = new HashMap<>(); | 
|---|
|  |  |  | Map<Integer, Integer>  mapDocInstallBedStatInfo = new HashMap<>(); | 
|---|
|  |  |  | mapCheckTypeVsReadyMax.keySet().forEach( checkType -> { | 
|---|
|  |  |  | mapOpeningCheckBedStatInfo.put(checkType, 0); | 
|---|
|  |  |  | mapOpeningInstallBedStatInfo.put(checkType, 0); | 
|---|
|  |  |  | mapOperatingCheckBedStatInfo.put(checkType, 0); | 
|---|
|  |  |  | mapOperatingInstallBedStatInfo.put(checkType, 0); | 
|---|
|  |  |  | mapDocCheckBedStatInfo.put(checkType, 0); | 
|---|
|  |  |  | mapDocInstallBedStatInfo.put(checkType, 0); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | //开通的 和 运营中 的工位统计 | 
|---|
|  |  |  | mapBedVsQueue.values().forEach( bedQueueBO -> { | 
|---|
|  |  |  | mapOpeningBed.values().forEach(bedQueueBO -> { | 
|---|
|  |  |  | Integer[] checkTypes = bedQueueBO.getCheckTypes(); | 
|---|
|  |  |  | Arrays.stream(checkTypes).forEach(checkType -> { | 
|---|
|  |  |  | if (bedQueueBO.opType == 0 || bedQueueBO.opType == 1) | 
|---|
|  |  |  | 
|---|
|  |  |  | else | 
|---|
|  |  |  | mapOpeningInstallBedStatInfo.put(checkType, mapOpeningInstallBedStatInfo.get(checkType) + 1 ); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (bedQueueBO.getStatus() == BedStatusEnum.DOCTOR_ON.getStatus()) { | 
|---|
|  |  |  | if (bedQueueBO.getStatus() == BedStatusEnum.DOCTOR_ON.getStatus() || | 
|---|
|  |  |  | bedQueueBO.getStatus() == BedStatusEnum.DOCTOR_PAUSE.getStatus()) { | 
|---|
|  |  |  | if (bedQueueBO.opType == 0 || bedQueueBO.opType == 1) | 
|---|
|  |  |  | mapOperatingCheckBedStatInfo.put(checkType, mapOperatingCheckBedStatInfo.get(checkType) + 1); | 
|---|
|  |  |  | mapDocCheckBedStatInfo.put(checkType, mapDocCheckBedStatInfo.get(checkType) + 1); | 
|---|
|  |  |  | else | 
|---|
|  |  |  | mapOperatingInstallBedStatInfo.put(checkType, mapOperatingInstallBedStatInfo.get(checkType) + 1); | 
|---|
|  |  |  | mapDocInstallBedStatInfo.put(checkType, mapDocInstallBedStatInfo.get(checkType) + 1); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // 接收..排队中患者..的优先队列 | 
|---|
|  |  |  | HashSet<BedQueueBO> hashSetOperatingBedQueue = new HashSet<>(); | 
|---|
|  |  |  | mapCheckTypePriorityQueue.values().forEach(priorityQueue->{ | 
|---|
|  |  |  | priorityQueue.stream().forEach(bedQueueBO -> hashSetOperatingBedQueue.add(bedQueueBO)); | 
|---|
|  |  |  | // 接收..排队中患者..的分流队列 | 
|---|
|  |  |  | HashSet<BedBO> hashSetOperatingBedQueue = new HashSet<>(); | 
|---|
|  |  |  | mapCheckTypeRoutingQueue.values().forEach(routingQueue->{ | 
|---|
|  |  |  | routingQueue.stream().forEach(bedQueueBO -> hashSetOperatingBedQueue.add(bedQueueBO)); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | 
|---|
|  |  |  | mapCheckTypeVsReadyMax.keySet().forEach( checkType -> { | 
|---|
|  |  |  | String str = mapOpeningCheckBedStatInfo.get(checkType) + " " + | 
|---|
|  |  |  | mapOpeningInstallBedStatInfo.get(checkType) + " " + | 
|---|
|  |  |  | mapOperatingCheckBedStatInfo.get(checkType) + " " + | 
|---|
|  |  |  | mapOperatingInstallBedStatInfo.get(checkType) + " "; | 
|---|
|  |  |  | mapDocCheckBedStatInfo.get(checkType) + " " + | 
|---|
|  |  |  | mapDocInstallBedStatInfo.get(checkType) + " "; | 
|---|
|  |  |  | mapCheckTypeBedInfo.put( checkType, str); | 
|---|
|  |  |  | }); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | monitorInfoVO.setCheckTypeBedInfo(mapCheckTypeBedInfo); | 
|---|
|  |  |  | monitorInfoVO.setQueueNum(mapBedVsQueue.size()); | 
|---|
|  |  |  | monitorInfoVO.setActiveQueueNum((int) mapBedVsQueue.values().stream().filter(item -> item.getStatus() == BedStatusEnum.DOCTOR_ON.getStatus()).count()); | 
|---|
|  |  |  | monitorInfoVO.setPriorityQueueNum( hashSetOperatingBedQueue.size() ); | 
|---|
|  |  |  | monitorInfoVO.setOpeningBedNum(mapOpeningBed.size()); | 
|---|
|  |  |  | monitorInfoVO.setDocBedNum((int) mapOpeningBed.values().stream().filter(item -> | 
|---|
|  |  |  | item.getStatus() == BedStatusEnum.DOCTOR_ON.getStatus() || | 
|---|
|  |  |  | item.getStatus() == BedStatusEnum.DOCTOR_PAUSE.getStatus()) | 
|---|
|  |  |  | .count()); | 
|---|
|  |  |  | monitorInfoVO.setRoutingBedNum( hashSetOperatingBedQueue.size() ); | 
|---|
|  |  |  | monitorInfoVO.setMapBedVsQueue(mapOpeningBed); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | } | 
|---|