diff --git a/app/Amqp/Consumer/AssignDoctorDelayDirectConsumer.php b/app/Amqp/Consumer/AssignDoctorDelayDirectConsumer.php new file mode 100644 index 0000000..e27953f --- /dev/null +++ b/app/Amqp/Consumer/AssignDoctorDelayDirectConsumer.php @@ -0,0 +1,328 @@ +info("开始:" . json_encode($data, JSON_UNESCAPED_UNICODE)); + + // 获取订单数据 + $params = array(); + $params['order_inquiry_id'] = $data['order_inquiry_id']; + $order_inquiry = OrderInquiry::getOne($params); + if (empty($order_inquiry)) { + Log::getInstance('queue-AssignDoctor')->error("错误:未查询到对应订单数据"); + return Result::DROP;// 销毁 + } + + // 检测订单状态 + $res = $this->checkInquiryStatus($order_inquiry); + if (!$res){ + Log::getInstance('queue-AssignDoctor')->error("错误:订单状态错误"); + return Result::DROP;// 销毁 + } + + // 检测分配时间 + $pay_time = strtotime($order_inquiry['pay_time']); + $diff_time = time() - $pay_time; + if ($diff_time < 0) { + Log::getInstance('queue-AssignDoctor')->error("错误:支付时间错误"); + return Result::DROP;// 销毁 + } + + try { + // 检测锁定状态 + $redis = $this->container->get(Redis::class); + $redis_key = "assign_octor_lock_" . $data['order_inquiry_id']; + $redis_lock = $redis->setnx($redis_key,1); + + // 设置过期时间-10s + $redis->expire($redis_key,10); + if (!$redis_lock){ + // 设置失败,表示已经设置该值 + Log::getInstance('queue-AssignDoctor')->info("锁定中,重回队列"); + return Result::REQUEUE; + } + }catch (\Exception $e){ + Log::getInstance('queue-AssignDoctor')->error("错误:" . $e->getMessage()); + return Result::ACK; + } + + Db::beginTransaction(); + try { + // 检测分配时间 + if ($diff_time > 300) { + Log::getInstance("queue-AssignDoctor")->info("信息:超出5分钟,执行退款"); + + $InquiryService = new InquiryService(); + + // 检测问诊订单执行退款次数 + Log::getInstance("queue-AssignDoctor")->info("信息:检测执行退款次数"); + $res = $InquiryService->checkInquiryRefundCount($order_inquiry['order_inquiry_id']); + if (!$res){ + Db::rollBack(); + Log::getInstance("queue-AssignDoctor")->error("错误:超出最大退款次数"); + return Result::ACK; + } + + Log::getInstance("queue-AssignDoctor")->info("信息:订单退款"); + $InquiryService->inquiryRefund($order_inquiry['order_inquiry_id'], "未分配到合适的医生"); + + Log::getInstance("queue-AssignDoctor")->info("信息:取消问诊订单"); + $save_data = array(); + $save_data['inquiry_status'] = 7; + $save_data['cancel_time'] = date("Y-m-d H:i:s", time()); + $save_data['cancel_reason'] = 3; // 取消订单原因(1:医生未接诊 2:主动取消 3:无可分配医生 4:客服取消 5:支付超时) + $save_data['cancel_remarks'] = "未分配到合适的医生"; // 取消订单备注 + $save_data['updated_at'] = date("Y-m-d H:i:s", time()); + + $params = array(); + $params['order_inquiry_id'] = $order_inquiry['order_inquiry_id']; + OrderInquiry::edit($params, $save_data); + + Log::getInstance("queue-AssignDoctor")->info("结束"); + Db::commit(); + return Result::ACK; + } + + // 检测执行次数 + $res = $this->checkHandleNumber($data['order_inquiry_id']); + if (!$res) { + // 超出执行次数后,不再进行分配,按照结束分配时间,重新加入队列。 + Log::getInstance("queue-AssignDoctor")->info("信息:超出最大执行次数或检测错误"); + + Log::getInstance("queue-AssignDoctor")->info("信息:重新加入延迟队列"); + + $queue_data = array(); + $queue_data['order_inquiry_id'] = $data['order_inquiry_id']; + + // 5分钟-支付时间-1s:10:00支付 此时10:04 5-(10:04-10:00) + $time = 1000 * (300- (time() - $pay_time) - 1); + + $message = new AssignDoctorDelayDirectProducer($queue_data); + $message->setDelayMs($time); + $producer = $this->container->get(Producer::class); + $res = $producer->produce($message); + if (!$res) { + Db::rollBack(); + Log::getInstance("queue-AssignDoctor")->error("错误:重新加入分配医生队列失败,重回队列"); + return Result::REQUEUE; + } + + Log::getInstance("queue-AssignDoctor")->info("结束"); + return Result::ACK; + } + + // 检测当前是否符合系统问诊时间 + $inquiryService = new InquiryService(); + $is_system_time_pass = $inquiryService->checkSystemInquiryTime($order_inquiry['inquiry_type']); + if (!$is_system_time_pass && $order_inquiry['inquiry_type'] == 4){ + // 非坐班时间 + Log::getInstance("queue-AssignDoctor")->info("信息:非坐班时间,执行退款"); + + $InquiryService = new InquiryService(); + + // 检测问诊订单执行退款次数 + Log::getInstance("queue-AssignDoctor")->info("信息:检测执行退款次数"); + $res = $InquiryService->checkInquiryRefundCount($order_inquiry['order_inquiry_id']); + if (!$res){ + Db::rollBack(); + Log::getInstance("queue-AssignDoctor")->error("错误:超出最大退款次数"); + return Result::ACK; + } + + Log::getInstance("queue-AssignDoctor")->info("信息:订单退款"); + $InquiryService->inquiryRefund($order_inquiry['order_inquiry_id'], "无可分配医生"); + + Log::getInstance("queue-AssignDoctor")->info("信息:取消问诊订单"); + $save_data = array(); + $save_data['inquiry_status'] = 7; + $save_data['cancel_time'] = date("Y-m-d H:i:s", time()); + $save_data['cancel_reason'] = 3; // 取消订单原因(1:医生未接诊 2:主动取消 3:无可分配医生 4:客服取消 5:支付超时) + $save_data['cancel_remarks'] = "未分配到合适的医生"; // 取消订单备注 + $save_data['updated_at'] = date("Y-m-d H:i:s", time()); + + $params = array(); + $params['order_inquiry_id'] = $order_inquiry['order_inquiry_id']; + OrderInquiry::edit($params, $save_data); + + Log::getInstance("queue-AssignDoctor")->info("结束"); + Db::commit(); + return Result::ACK; + } + + Log::getInstance("queue-AssignDoctor")->info("信息:分配医生"); + $UserDoctorService = new UserDoctorService(); + $doctor_id = $UserDoctorService->getInquiryAssignDoctor($order_inquiry['inquiry_type'],$order_inquiry['patient_id'],$is_system_time_pass); + if (empty($doctor_id)){ + Log::getInstance("queue-AssignDoctor")->info("信息:无合适医生"); + Db::rollBack(); + + Log::getInstance("queue-AssignDoctor")->info("信息:重新加入延迟队列"); + + $queue_data = array(); + $queue_data['order_inquiry_id'] = $data['order_inquiry_id']; + + $message = new AssignDoctorDelayDirectProducer($queue_data); + $message->setDelayMs(1000 * 60); + $producer = $this->container->get(Producer::class); + $res = $producer->produce($message); + if (!$res) { + Db::rollBack(); + Log::getInstance("queue-AssignDoctor")->error("错误:重新加入分配医生队列失败,重回队列"); + return Result::REQUEUE; + } + + Log::getInstance("queue-AssignDoctor")->info("结束"); + return Result::ACK; + } + + // 更改数据库 + $save_data = array(); + $save_data['doctor_id'] = $doctor_id; + $save_data['inquiry_status'] = 3; // 待接诊 + + $params = array(); + $params['order_inquiry_id'] = $order_inquiry['order_inquiry_id']; + OrderInquiry::edit($params, $save_data); + + // 加入未接诊取消订单延迟队列 + $data = array(); + $data['order_inquiry_id'] = $order_inquiry['order_inquiry_id']; + $message = new CancelUnInquiryOrdersDelayDirectProducer($data); + // 快速/购药-5分钟 + $message->setDelayMs(1000 * 60 * 2); + $producer = $this->container->get(Producer::class); + $res = $producer->produce($message); + if (!$res) { + Db::rollBack(); + Log::getInstance("queue-AssignDoctor")->error("加入未接诊取消订单延迟队列失败"); + return Result::ACK; + } + + Db::commit(); + Log::getInstance("queue-AssignDoctor")->info("成功"); + } catch (\Exception $e) { + Db::rollBack(); + Log::getInstance("queue-AssignDoctor")->error("失败:" . $e->getMessage()); + return Result::REQUEUE; // 重回队列 + } + + return Result::ACK; + } + + /** + * 检测订单状态 + * @param array|object $order_inquiry + * @return bool + */ + private function checkInquiryStatus(array|object $order_inquiry): bool + { + // 检测订单分配状态 + if (!empty($order_inquiry['doctor_id'])) { + Log::getInstance()->error("分配医生队列执行失败:已分配医生"); + return false; + } + + // 检测订单类型 + if ($order_inquiry['inquiry_type'] != 2 && $order_inquiry['inquiry_type'] != 4) { + Log::getInstance()->error("分配医生队列执行失败:订单非快速问诊、问诊购药类型"); + return false; + } + + // 检测订单退款状态 + if (in_array($order_inquiry['inquiry_refund_status'], [1, 2, 3])) { + // 问诊订单退款状态(0:无退款 1:申请退款 2:退款中 3:退款成功 4:拒绝退款 5:退款关闭) + Log::getInstance()->error("分配医生队列执行失败:订单存在退款"); + return false; + } + + // 检测订单状态 + if ($order_inquiry['inquiry_status'] != 2) { + // 问诊订单状态(1:待支付 2:待分配 3:待接诊 4:已接诊 5:已完成 6:已结束 7:已取消) + Log::getInstance()->error("分配医生队列执行失败:订单状态错误:当前为" . $order_inquiry['inquiry_status'] . " 无法进行分配"); + return false; + } + + // 检测订单支付状态 + if ($order_inquiry['inquiry_pay_status'] != 2) { + // 支付状态(1:未支付 2:已支付 3:支付中 4:支付失败 5:支付超时 6:支付关闭 7:已撤销 8:转入退款) + Log::getInstance()->error("分配医生队列执行失败:订单支付状态错误:当前为" . $order_inquiry['inquiry_pay_status'] . " 无法进行分配"); + return false; + } + + return true; + } + + /** + * 检测执行次数 + * @param string $order_inquiry_id + * @return bool + * @throws ContainerExceptionInterface + * @throws NotFoundExceptionInterface + */ + protected function checkHandleNumber(string $order_inquiry_id): bool + { + try { + $redis = $this->container->get(Redis::class); + + $redis_key = "assign_doctor_number_" . $order_inquiry_id; + $redis_value = $redis->get($redis_key); + if (empty($redis_value)) { + $redis->set($redis_key, 1, 60 * 60 * 24 * 5); + return true; + } + + // 执行次数过多 + if ($redis_value > 3) { + // 加入短信队列,通知管理员 + + return false; + } + + $redis->incr($redis_key); + } catch (\Exception $e) { + return false; + } + + return true; + } + +} diff --git a/app/Amqp/Producer/AssignDoctorDelayDirectProducer.php b/app/Amqp/Producer/AssignDoctorDelayDirectProducer.php new file mode 100644 index 0000000..dc70c25 --- /dev/null +++ b/app/Amqp/Producer/AssignDoctorDelayDirectProducer.php @@ -0,0 +1,27 @@ +payload = $data; + } +}