新增分配医生延迟队列

This commit is contained in:
wucongxing 2023-04-26 15:35:30 +08:00
parent d746a7ae8a
commit 9636f0ce34
2 changed files with 355 additions and 0 deletions

View File

@ -0,0 +1,328 @@
<?php
declare(strict_types=1);
namespace App\Amqp\Consumer;
use App\Amqp\Producer\AssignDoctorDelayDirectProducer;
use App\Amqp\Producer\CancelUnInquiryOrdersDelayDirectProducer;
use App\Amqp\Producer\UserCouponExpiredDelayDirectProducer;
use App\Constants\HttpEnumCode;
use App\Model\OrderInquiry;
use App\Services\InquiryService;
use App\Services\UserDoctorService;
use App\Utils\Log;
use Hyperf\Amqp\Message\ConsumerDelayedMessageTrait;
use Hyperf\Amqp\Message\ProducerDelayedMessageTrait;
use Hyperf\Amqp\Message\Type;
use Hyperf\Amqp\Producer;
use Hyperf\Amqp\Result;
use Hyperf\Amqp\Annotation\Consumer;
use Hyperf\Amqp\Message\ConsumerMessage;
use Hyperf\DbConnection\Db;
use Hyperf\Redis\Redis;
use PhpAmqpLib\Message\AMQPMessage;
use Psr\Container\ContainerExceptionInterface;
use Psr\Container\NotFoundExceptionInterface;
#[Consumer(nums: 1)]
class AssignDoctorDelayDirectConsumer extends ConsumerMessage
{
use ProducerDelayedMessageTrait;
use ConsumerDelayedMessageTrait;
protected string $exchange = 'amqp.delay.direct';
protected ?string $queue = 'assign.doctor.delay.queue';
protected string $type = Type::DIRECT; //Type::FANOUT;
protected string|array $routingKey = 'AssignDoctor';
public function consumeMessage($data, AMQPMessage $message): string
{
Log::getInstance('queue-AssignDoctor')->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分钟-支付时间-1s1000支付 此时1004 5-1004-1000
$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;
}
}

View File

@ -0,0 +1,27 @@
<?php
declare(strict_types=1);
namespace App\Amqp\Producer;
use Hyperf\Amqp\Annotation\Producer;
use Hyperf\Amqp\Message\ProducerDelayedMessageTrait;
use Hyperf\Amqp\Message\ProducerMessage;
use Hyperf\Amqp\Message\Type;
#[Producer]
class AssignDoctorDelayDirectProducer extends ProducerMessage
{
use ProducerDelayedMessageTrait;
protected string $exchange = 'amqp.delay.direct';
protected string $type = Type::DIRECT;
protected string|array $routingKey = 'AssignDoctor';
public function __construct($data)
{
$this->payload = $data;
}
}