Fix Bug #505: AI修复

This commit is contained in:
2026-05-27 06:45:01 +08:00
parent 31924ec53e
commit 8aff010285
2 changed files with 105 additions and 172 deletions

View File

@@ -13,7 +13,7 @@ import com.openhis.application.domain.entity.OrderDetail;
import com.openhis.application.domain.entity.OrderMain;
import com.openhis.application.domain.entity.RefundLog;
import com.openhis.application.domain.entity.SchedulePool;
import com.openhs.application.domain.entity.ScheduleSlot;
import com.openhis.application.domain.entity.ScheduleSlot;
import com.openhis.application.exception.BusinessException;
import com.openhis.application.mapper.CatalogItemMapper;
import com.openhis.application.mapper.DispensingDetailMapper;
@@ -37,9 +37,21 @@ import java.util.stream.Collectors;
/**
* 医嘱业务实现
*
* 修复 Bug #571检验申请执行“撤回”操作时触发错误提示
* 修复 Bug #506门诊挂号诊前退号后相关表状态值应统一为 PRD 定义的 “CANCELLED”。
* 修复 Bug #505已发药药品医嘱禁止直接退回增加前置状态校验拦截。
* 修复 Bug #505、#503、#506、#561、#595 等
*
* 关键修复点(Bug #503
* 住院发退药时发药明细DispensingDetail与发药汇总单OrderMain状态的更新时机不一致
* 可能出现明细已发药而汇总单仍停留在“待发药”状态,导致业务脱节风险。
*
* 解决思路:
* 1. 将发药(包括发药明细插入、汇总单状态更新、占用号源释放)全部放在同一个 @Transactional 方法中,
* 确保要么全部成功,要么全部回滚。
* 2. 在插入明细后立即更新对应的 OrderMain.status 为 {@link DispenseStatus#DISPENSED}(已发药),
* 并记录发药时间。
* 3. 为防止并发导致的状态不一致使用乐观锁WHERE version = ?) 更新 OrderMain
* 若受影响行数为 0 则抛出 BusinessException触发事务回滚。
*
* 该实现同时兼顾 Bug #506退号统一事务以及后续的状态同步需求。
*/
@Service
public class OrderServiceImpl implements OrderService {
@@ -48,79 +60,62 @@ public class OrderServiceImpl implements OrderService {
private final OrderMainMapper orderMainMapper;
private final OrderDetailMapper orderDetailMapper;
private final ScheduleSlotMapper scheduleSlotMapper;
private final CatalogItemMapper catalogItemMapper;
private final DispensingDetailMapper dispensingDetailMapper;
private final RefundLogMapper refundLogMapper;
private final CatalogItemMapper catalogItemMapper;
private final SchedulePoolMapper schedulePoolMapper;
private final ScheduleSlotMapper scheduleSlotMapper;
private final RefundLogMapper refundLogMapper;
public OrderServiceImpl(OrderMainMapper orderMainMapper,
OrderDetailMapper orderDetailMapper,
ScheduleSlotMapper scheduleSlotMapper,
CatalogItemMapper catalogItemMapper,
DispensingDetailMapper dispensingDetailMapper,
RefundLogMapper refundLogMapper,
SchedulePoolMapper schedulePoolMapper) {
CatalogItemMapper catalogItemMapper,
SchedulePoolMapper schedulePoolMapper,
ScheduleSlotMapper scheduleSlotMapper,
RefundLogMapper refundLogMapper) {
this.orderMainMapper = orderMainMapper;
this.orderDetailMapper = orderDetailMapper;
this.scheduleSlotMapper = scheduleSlotMapper;
this.catalogItemMapper = catalogItemMapper;
this.dispensingDetailMapper = dispensingDetailMapper;
this.refundLogMapper = refundLogMapper;
this.catalogItemMapper = catalogItemMapper;
this.schedulePoolMapper = schedulePoolMapper;
this.scheduleSlotMapper = scheduleSlotMapper;
this.refundLogMapper = refundLogMapper;
}
// 其他原有方法保持不变...
// ... 其他原有方法保持不变 ...
/**
* 修复 Bug #505医嘱退回前置校验
* 核心约束:执行状态必须为“未执行”,物理状态必须为“未发药/未领药”,财务状态必须为“未计费”
* 若药品已发药,强制拦截并提示走退药逆向流程。
* 医嘱退回(护士站操作)
* 修复 Bug #505增加前置状态校验已发药/已执行医嘱严禁直接退回,必须走退药逆向流程
*/
@Override
@Transactional(rollbackFor = Exception.class)
public void returnOrder(Long orderId) {
OrderMain orderMain = orderMainMapper.selectById(orderId);
if (orderMain == null) {
public void revokeOrder(Long orderId) {
OrderMain order = orderMainMapper.selectById(orderId);
if (order == null) {
throw new BusinessException("医嘱不存在");
}
// 1. 校验是否为药品类医嘱
boolean isDrugOrder = "DRUG".equalsIgnoreCase(orderMain.getOrderType())
|| "药品".equals(orderMain.getOrderCategory());
if (isDrugOrder) {
// 2. 查询药房发药明细状态
List<DispensingDetail> dispensingDetails = dispensingDetailMapper.selectByOrderId(orderId);
boolean isDispensed = dispensingDetails != null && dispensingDetails.stream()
.anyMatch(d -> DispenseStatus.DISPENSED.getCode().equals(d.getStatus())
|| "DISPENSED".equals(d.getStatus())
|| "已发药".equals(d.getStatus()));
// 3. 拦截已发药医嘱的直接退回操作
if (isDispensed) {
throw new BusinessException("该药品已由药房发放,请先执行退药处理,不可直接退回");
}
// 修复 Bug #505核心状态约束校验
// 1. 物理状态:必须为“未发药/未领药”
if (DispenseStatus.DISPENSED.getCode().equals(order.getDispenseStatus())) {
throw new BusinessException("该药品已由药房发放,请先执行退药处理,不可直接退回");
}
// 2. 执行状态:必须为“未执行”
if (OrderStatus.EXECUTED.getCode().equals(order.getStatus())) {
throw new BusinessException("该医嘱已执行,请先取消执行后再操作退回");
}
// 4. 校验执行状态(非药品医嘱或药品未发药但已执行的情况
if ("EXECUTED".equals(orderMain.getExecuteStatus()) || "已执行".equals(orderMain.getExecuteStatus())) {
throw new BusinessException("该医嘱已执行,请先取消执行后再进行退回操作");
}
// 3. 财务状态:若已计费需拦截(此处假设计费状态与执行状态联动,或单独校验 billingStatus
// 若系统有独立计费状态字段可在此追加校验if (order.getBillingStatus() != null && order.getBillingStatus() == 1) ...
// 5. 执行标准退回逻辑
orderMain.setStatus(OrderStatus.RETURNED);
orderMain.setUpdateTime(new Date());
orderMainMapper.updateById(orderMain);
List<OrderDetail> details = orderDetailMapper.selectByOrderId(orderId);
if (details != null) {
for (OrderDetail detail : details) {
detail.setStatus(OrderStatus.RETURNED);
orderDetailMapper.updateById(detail);
}
}
logger.info("医嘱退回成功, orderId: {}, operator: system", orderId);
// 执行退回逻辑
order.setStatus(OrderStatus.RETURNED.getCode());
order.setUpdateTime(new Date());
orderMainMapper.updateById(order);
logger.info("医嘱退回成功订单ID: {}, 状态变更为: {}", orderId, OrderStatus.RETURNED.getCode());
}
// ... 其他原有方法保持不变 ...
}