NativeQrcodePaymentCmd.java 5.78 KB
package com.java110.acct.cmd.payment;

import com.alibaba.fastjson.JSONObject;
import com.java110.acct.payment.IPaymentBusiness;
import com.java110.core.annotation.Java110Cmd;
import com.java110.core.context.CmdContextUtils;
import com.java110.core.context.ICmdDataFlowContext;
import com.java110.core.event.cmd.Cmd;
import com.java110.core.event.cmd.CmdEvent;
import com.java110.core.factory.GenerateCodeFactory;
import com.java110.core.log.LoggerFactory;
import com.java110.dto.payment.PaymentOrderDto;
import com.java110.intf.acct.IPaymentPoolConfigV1InnerServiceSMO;
import com.java110.intf.acct.IPaymentPoolV1InnerServiceSMO;
import com.java110.intf.fee.IPayFeeV1InnerServiceSMO;
import com.java110.utils.cache.CommonCache;
import com.java110.utils.cache.UrlCache;
import com.java110.utils.exception.CmdException;
import com.java110.utils.factory.ApplicationContextFactory;
import com.java110.utils.util.Assert;
import com.java110.vo.ResultVo;
import org.slf4j.Logger;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.ResponseEntity;

import java.text.ParseException;

/**
 * 原生二维码支付命令类
 * 负责处理原生二维码支付业务,包括参数验证、支付订单创建和支付二维码生成
 * 通过指定的支付业务类型创建支付订单,并生成对应的支付二维码URL
 * 
 * @author wuxw
 * @version 1.0
 * @since 2023-06-25
 */
@Java110Cmd(serviceCode = "payment.nativeQrcodePayment")
public class NativeQrcodePaymentCmd extends Cmd {

    /**
     * 日志记录器
     */
    private static final Logger logger = LoggerFactory.getLogger(NativeQrcodePaymentCmd.class);

    /**
     * 默认支付适配器名称 - 微信原生二维码支付工厂
     */
    protected static final String DEFAULT_PAYMENT_ADAPT = "wechatNativeQrcodePaymentFactory";

    /**
     * 支付池配置服务接口
     */
    @Autowired
    private IPaymentPoolConfigV1InnerServiceSMO paymentPoolConfigV1InnerServiceSMOImpl;

    /**
     * 支付池服务接口
     */
    @Autowired
    private IPaymentPoolV1InnerServiceSMO paymentPoolV1InnerServiceSMOImpl;

    /**
     * 缴费服务接口
     */
    @Autowired
    private IPayFeeV1InnerServiceSMO payFeeV1InnerServiceSMOImpl;

    /**
     * 参数验证方法
     * 验证请求参数中是否包含必要的业务参数和项目信息
     *
     * @param event   命令事件对象,包含请求相关信息
     * @param context 命令数据流上下文,包含请求和响应数据
     * @param reqJson 请求参数JSON对象
     * @throws CmdException   当参数验证失败时抛出命令异常
     * @throws ParseException 当参数解析失败时抛出解析异常
     */
    @Override
    public void validate(CmdEvent event, ICmdDataFlowContext context, JSONObject reqJson) throws CmdException, ParseException {
        // 验证请求参数中必须包含business字段
        Assert.hasKeyAndValue(reqJson, "business", "未包含业务");
        // 验证请求参数中必须包含communityId字段
        Assert.hasKeyAndValue(reqJson, "communityId", "未包含项目");
    }

    /**
     * 执行支付命令
     * 处理原生二维码支付业务逻辑,包括创建支付订单、生成支付二维码等
     *
     * @param event   命令事件对象
     * @param context 命令数据流上下文
     * @param reqJson 请求参数JSON对象
     * @throws CmdException   当支付业务处理失败时抛出命令异常
     * @throws ParseException 当数据解析失败时抛出解析异常
     */
    @Override
    public void doCmd(CmdEvent event, ICmdDataFlowContext context, JSONObject reqJson) throws CmdException, ParseException {

        // 记录支付请求参数日志
        logger.debug(">>>>>>>>>>>>>>>>支付参数报文,{}", reqJson.toJSONString());
        
        // 从请求头中获取应用ID和用户ID
        String appId = context.getReqHeaders().get("app-id");
        String userId = context.getReqHeaders().get("user-id");
        
        // 将用户ID和店铺ID添加到请求参数中
        reqJson.put("createUserId", userId);
        reqJson.put("storeId", CmdContextUtils.getStoreId(context));

        // 1.0 根据业务类型获取对应的支付业务处理器
        IPaymentBusiness paymentBusiness = ApplicationContextFactory.getBean(reqJson.getString("business"), IPaymentBusiness.class);

        // 检查支付业务处理器是否存在
        if (paymentBusiness == null) {
            throw new CmdException("当前支付业务不支持");
        }

        // 2.0 调用支付业务处理器创建统一支付订单
        PaymentOrderDto paymentOrderDto = paymentBusiness.unified(context, reqJson);
        
        // 设置支付订单的应用ID和用户ID
        paymentOrderDto.setAppId(appId);
        paymentOrderDto.setUserId(userId);
        
        // 将支付金额添加到请求参数中
        reqJson.put("money", paymentOrderDto.getMoney());

        // 生成唯一的支付令牌
        String token = GenerateCodeFactory.getUUID();

        // 将支付请求参数缓存到Redis中,使用支付令牌作为key
        CommonCache.setValue("nativeQrcodePayment_" + token, reqJson.toJSONString(), CommonCache.PAY_DEFAULT_EXPIRE_TIME);
        
        // 创建响应结果对象
        JSONObject result = new JSONObject();
        
        // 生成支付二维码URL,包含支付令牌参数
        result.put("codeUrl", UrlCache.getOwnerUrl() + "/#/pages/fee/qrCodeCashier?qrToken=" + token);

        // 创建HTTP响应实体
        ResponseEntity<String> responseEntity = ResultVo.createResponseEntity(result);

        // 记录支付厂家返回结果日志
        logger.debug("调用支付厂家返回,{}", responseEntity);
        
        // 设置响应实体到上下文中
        context.setResponseEntity(responseEntity);

    }

}