宋子宪博客

SpringBoot对接银联支付系统

首先需要到银联注册帐号,然后下载需要的证书,如图

然后下载jdkhttps://open.unionpay.com/tjweb/acproduct/list?apiSvcId=448&index=5
把SDK中的com.unionpay.acp.sdk包下的类复制到开发项目中

把SDK中的com.unionpay.acp.demo包下的DemoBase类复制到开发项目com.unionpay.acp.sdk包中,改名为UnionpayBese
把src下的配置文件复制到项目中

打开acp_sdk.properties配置修改证书路径地址

编写支付共同行为接口

public interface PayStrategyService {

    /**
     * 共同算法实现
     */
    public String toPayHtml(PaymentChannelEntity pymentChannel, PayMentTransacDTO payMentTransacDTO);
}

编写支付共同行为实现类

@Slf4j
public class UnionPayStrategyServiceImpl implements PayStrategyService {

    @Override
    public String toPayHtml(PaymentChannelEntity paymentChannel, PayMentTransacDTO payMentTransacDTO) {
        log.info(">>>>>>>>银联支付组装参数开始<<<<<<<<<<<<");

        Map<String, String> requestData = new HashMap<String, String>();

        /*** 银联全渠道系统,产品参数,除了encoding自行选择外其他不需修改 ***/
        requestData.put("version", UnionPayBase.version); // 版本号,全渠道默认值
        requestData.put("encoding", UnionPayBase.encoding); // 字符集编码,可以使用UTF-8,GBK两种方式
        requestData.put("signMethod", SDKConfig.getConfig().getSignMethod()); // 签名方法
        requestData.put("txnType", "01"); // 交易类型 ,01:消费
        requestData.put("txnSubType", "01"); // 交易子类型, 01:自助消费
        requestData.put("bizType", "000201"); // 业务类型,B2C网关支付,手机wap支付
        requestData.put("channelType", "07"); // 渠道类型,这个字段区分B2C网关支付和手机wap支付;07:PC,平板
        // 08:手机

        /*** 商户接入参数 ***/
        String merchantId = paymentChannel.getMerchantId();
        requestData.put("merId", merchantId); // 商户号码,请改成自己申请的正式商户号或者open上注册得来的777测试商户号
        requestData.put("accessType", "0"); // 接入类型,0:直连商户
        String paymentId = payMentTransacDTO.getPaymentId();
        // 在微服务电商项目中 订单系统(orderId)   支付系统 支付id
        requestData.put("orderId", paymentId); // 商户订单号,8-40位数字字母,不能含“-”或“_”,可以自行定制规则
        requestData.put("txnTime", format(payMentTransacDTO.getCreatedTime())); // 订单发送时间,取系统时间,格式为YYYYMMDDhhmmss,必须取当前时间,否则会报txnTime无效
        requestData.put("currencyCode", "156"); // 交易币种(境内商户一般是156 人民币)
        Long payAmount = payMentTransacDTO.getPayAmount();
        requestData.put("txnAmt", payAmount + ""); // 交易金额,单位分,不要带小数点
        // requestData.put("reqReserved", "透传字段");
        // //请求方保留域,如需使用请启用即可;透传字段(可以实现商户自定义参数的追踪)本交易的后台通知,对本交易的交易状态查询交易、对账文件中均会原样返回,商户可以按需上传,长度为1-1024个字节。出现&={}[]符号时可能导致查询接口应答报文解析失败,建议尽量只传字母数字并使用|分割,或者可以最外层做一次base64编码(base64编码之后出现的等号不会导致解析失败可以不用管)。

        //商品名称 或者商品描述
        requestData.put("riskRateInfo", "{commodityName=测试商品名称}");

        //
        // 前台通知地址 (需设置为外网能访问 http https均可),支付成功后的页面 点击“返回商户”按钮的时候将异步通知报文post到该地址
        // 如果想要实现过几秒中自动跳转回商户页面权限,需联系银联业务申请开通自动返回商户权限
        // 异步通知参数详见open.unionpay.com帮助中心 下载 产品接口规范 网关支付产品接口规范 消费交易 商户通知\
        String syncUrl = paymentChannel.getSyncUrl();
        requestData.put("frontUrl", syncUrl);

        // 后台通知地址(需设置为【外网】能访问 http
        // https均可),支付成功后银联会自动将异步通知报文post到商户上送的该地址,失败的交易银联不会发送后台通知
        // 后台通知参数详见open.unionpay.com帮助中心 下载 产品接口规范 网关支付产品接口规范 消费交易 商户通知
        // 注意:1.需设置为外网能访问,否则收不到通知 2.http https均可 3.收单后台通知后需要10秒内返回http200或302状态码
        // 4.如果银联通知服务器发送通知后10秒内未收到返回状态码或者应答码非http200,那么银联会间隔一段时间再次发送。总共发送5次,每次的间隔时间为0,1,2,4分钟。
        // 5.后台通知地址如果上送了带有?的参数,例如:http://abc/web?a=b&c=d
        // 在后台通知处理程序验证签名之前需要编写逻辑将这些字段去掉再验签,否则将会验签失败
        String asynUrl = paymentChannel.getAsynUrl();
        requestData.put("backUrl", asynUrl);

        // 订单超时时间。
        // 超过此时间后,除网银交易外,其他交易银联系统会拒绝受理,提示超时。
        // 跳转银行网银交易如果超时后交易成功,会自动退款,大约5个工作日金额返还到持卡人账户。
        // 此时间建议取支付时的北京时间加15分钟。
        // 超过超时时间调查询接口应答origRespCode不是A6或者00的就可以判断为失败。
        requestData.put("payTimeout",
                new SimpleDateFormat("yyyyMMddHHmmss").format(new Date().getTime() + 15 * 60 * 1000));

        //////////////////////////////////////////////////
        //
        // 报文中特殊用法请查看 PCwap网关跳转支付特殊用法.txt
        //
        //////////////////////////////////////////////////

        /** 请求参数设置完毕,以下对请求参数进行签名并生成html表单,将表单写入浏览器跳转打开银联页面 **/
        Map<String, String> submitFromData = AcpService.sign(requestData, UnionPayBase.encoding); // 报文中certId,signature的值是在signData方法中获取并自动赋值的,只要证书配置正确即可。

        String requestFrontUrl = SDKConfig.getConfig().getFrontRequestUrl(); // 获取请求银联的前台地址:对应属性文件acp_sdk.properties文件中的acpsdk.frontTransUrl
        String html = AcpService.createAutoFormHtml(requestFrontUrl, submitFromData, UnionPayBase.encoding); // 生成自动跳转的Html表单

        LogUtil.writeLog("打印请求HTML,此为请求报文,为联调排查问题的依据:" + html);
        // 将生成的html写到浏览器中完成自动跳转打开银联支付页面;这里调用signData之后,将html写到浏览器跳转到银联页面之前均不能对html中的表单项的名称和值进行修改,如果修改会导致验签不通过
        return html;
    }

    private String format(Date timeDate) {
        String date = new java.text.SimpleDateFormat("yyyyMMddHHmmss").format(timeDate);
        return date;
    }
}

在szx-shop-unionpay-plugin中创建com.songzixian.pay.init包,创建InitUnionPayProject类,让项目启动时自动加载证书

@Component
public class InitUnionPayProject implements ApplicationRunner {

    // springboot 项目启动的时候 执行该方法
    @Override
    public void run(ApplicationArguments args) throws Exception {
        SDKConfig.getConfig().loadPropertiesFromSrc();
    }
}

证书加载完毕启动成功

访问localhost:8600/toPayHtml?chnnelId=yinlian_pay&payToken=pay_607d328eb99a4655a45652929ed68d41
返回一个thml

复制下了,把\"全部替换为",新建一个html文档放进去,访问,打开跳转到支付页面

测试账号信息:
卡号:6216261000000000018
姓名:全渠道
身份证号码: 341126197709218366

提交支付

当前页面是本站的「Google AMP」版。查看和发表评论请点击:完整版 »