前言
虽然对接支付宝支付功能很常见,但是随着项目业务不断调整,有时候支付宝也在迭代。过段时间不看很可能就忘了,比如证书过期了,连密钥生成都不知道怎么弄。所以我就在这里就再总结一下JSAPI的对接步骤和遇到的几个小坑。
报错
- 服务端请求接口返回 “商户协议状态非正常状态”。
- 前端调起支付时报6001错误。
- 最新的JSAPI已经不再支持预支付串。
接口加密方式
根据我以往的经验,对接支付宝支付,偶尔改动的地方是返回部分,而容易忘记的是接口加密部分。
密钥方式
用SDK要使用AopClient包,要求有开发者私钥(APP_PRIVATE_KEY)和支付宝公钥(ALIPAY_PUBLIC_KEY),都是在支付宝商户平台操作生成。
证书方式
证书模式用的是AopCertClient包,要求在商户平台下载应用公钥证书、支付宝公钥证书、支付宝根证书。

编码
配置文件
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| <?php
return [ "GATEWAY_URL" => 'https://openapi.alipay.com/gateway.do', "APP_ID" => "20180**********15", "PID" => '', "RSA_PRIVATE_KEY" => 'ali_private.pem', "ALIPAYRSA_PUBLIC_KEY" => 'apiclient_cert.pem', ];
|
工厂类
1 2 3 4 5 6 7 8 9 10 11 12
| <?php
namespace pay;
class PayFactory { public static function factory($class_name) { $class_name = 'pay\\'.$class_name; return new $class_name; } }
|
支付方法抽象
1 2 3 4 5 6 7 8 9 10
| <?php
namespace pay;
abstract class PayBase { abstract public function pay($order,$notify_url); abstract public function batchpay($order); abstract public function version(); }
|
密钥方式封装
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52
| <?php
namespace pay;
use think\facade\Env;
require_once 'lib/aop/AopCertClient.php'; require_once 'lib/aop/request/AlipayFundTransUniTransferRequest.php'; require_once 'lib/aop/request/AlipayOpenPublicTemplateMessageIndustryModifyRequest.php'; require_once 'lib/aop/request/AlipayUserInfoShareRequest.php'; require_once 'lib/aop/request/AlipaySystemOauthTokenRequest.php'; require_once 'lib/aop/request/AlipayTradeAppPayRequest.php'; require_once 'lib/aop/request/AlipayTradeCreateRequest.php'; require_once 'lib/aop/request/AlipayFundAuthOrderAppFreezeRequest.php'; require_once 'lib/aop/request/AlipayFundAuthOrderUnfreezeRequest.php'; require_once 'lib/aop/request/AlipayTradeRefundRequest.php';
class AliPay extends PayBase { private $config = [];
private $aop = null; public function __construct() { $this->config = include 'config/ali_pay_config.php';
$path = dirname(__FILE__);
$RSA_PRIVATE_KEY = file_get_contents($path.DIRECTORY_SEPARATOR."keys".DIRECTORY_SEPARATOR.$this->config['RSA_PRIVATE_KEY']);
$ALIPAYRSA_PUBLIC_KEY = file_get_contents($path.DIRECTORY_SEPARATOR."keys".DIRECTORY_SEPARATOR.$this->config['ALIPAYRSA_PUBLIC_KEY']);
$this->aop = new \AopClient();
$this->aop->gatewayUrl = $this->config["GATEWAY_URL"]; $this->aop->appId = $this->config["APP_ID"]; $this->aop->rsaPrivateKey = $RSA_PRIVATE_KEY; $this->aop->alipayrsaPublicKey= $ALIPAYRSA_PUBLIC_KEY; $this->aop->apiVersion = "1.0"; $this->aop->signType = "RSA2"; $this->aop->postCharset = "UTF-8"; $this->aop->format = "json"; }
public function version() { return "1.0"; }
|
证书方式封装
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61
| <?php
namespace pay;
use think\facade\Env;
require_once 'lib/aop/AopCertClient.php'; require_once 'lib/aop/request/AlipayFundTransUniTransferRequest.php'; require_once 'lib/aop/request/AlipayOpenPublicTemplateMessageIndustryModifyRequest.php'; require_once 'lib/aop/request/AlipayUserInfoShareRequest.php'; require_once 'lib/aop/request/AlipaySystemOauthTokenRequest.php'; require_once 'lib/aop/request/AlipayTradeAppPayRequest.php'; require_once 'lib/aop/request/AlipayTradeCreateRequest.php'; require_once 'lib/aop/request/AlipayFundAuthOrderAppFreezeRequest.php'; require_once 'lib/aop/request/AlipayFundAuthOrderUnfreezeRequest.php'; require_once 'lib/aop/request/AlipayTradeRefundRequest.php';
class AliPay extends PayBase { private $config = [];
private $aop = null; public function __construct() { $this->config = include 'config/ali_pay_config.php';
$path = dirname(__FILE__);
$RSA_PRIVATE_KEY = file_get_contents($path.DIRECTORY_SEPARATOR."keys".DIRECTORY_SEPARATOR.$this->config['RSA_PRIVATE_KEY']);
$ALIPAYRSA_PUBLIC_KEY = file_get_contents($path.DIRECTORY_SEPARATOR."keys".DIRECTORY_SEPARATOR.$this->config['ALIPAYRSA_PUBLIC_KEY']);
$this->aop = new \AopCertClient ();
$appCertPath = $path.DIRECTORY_SEPARATOR."keys".DIRECTORY_SEPARATOR."cert".DIRECTORY_SEPARATOR."appCertPublicKey_2018091961451345.crt"; $alipayCertPath = $path.DIRECTORY_SEPARATOR."keys".DIRECTORY_SEPARATOR."cert".DIRECTORY_SEPARATOR."alipayCertPublicKey_RSA2.crt"; $rootCertPath = $path.DIRECTORY_SEPARATOR."keys".DIRECTORY_SEPARATOR."cert".DIRECTORY_SEPARATOR."alipayRootCert.crt";
$this->aop->gatewayUrl = $this->config["GATEWAY_URL"]; $this->aop->appId = $this->config["APP_ID"]; $this->aop->rsaPrivateKey = $RSA_PRIVATE_KEY; $this->aop->apiVersion = "1.0"; $this->aop->signType = "RSA2"; $this->aop->postCharset = "UTF-8"; $this->aop->format = "json";
$this->aop->alipayrsaPublicKey = $this->aop->getPublicKey($alipayCertPath); $this->aop->isCheckAlipayPublicCert = true; $this->aop->appCertSN = $this->aop->getCertSN($appCertPath); $this->aop->alipayRootCertSN = $this->aop->getRootCertSN($rootCertPath);
}
|
支付封装
这里要特别注意的是调用执行方法时,证书模式execute,密钥模式sdkExecute。密钥模式返回orderStr,证书模式返回tradeNO。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90
| public function pay($order,$notify_url) { $data = $this->make($order);
$timeout_express = "30m"; $total_amount = $data['total_amount']; $subject = $data['subject']; $out_trade_no = $data['out_trade_no'];
$content = [ "timeout_express" => "30m", "total_amount" => $total_amount, "subject" => $subject, "out_trade_no" => $out_trade_no, "product_code" => "JSAPI_PAY", "op_app_id" => $this->config["APP_ID"], "buyer_id" => $order['open_id'], ];
$jsonData = json_encode($content,JSON_UNESCAPED_UNICODE);
$request = new \AlipayTradeCreateRequest();
$request->setNotifyUrl(Env::get('pay.coupon_notify')); $request->setBizContent($jsonData); $result = $this->aop->execute($request);
$responseApiName = str_replace(".","_",$request->getApiMethodName())."_response"; $response = $result->$responseApiName;
if(!empty($response->code)&&$response->code==10000) { return $response->trade_no; } else{ throw new \Exception('调用JSAPI支付接口失败~'); } }
private function make($order,$type="pay") { $data = [];
if ($type == "pay") { $data['total_amount'] = $order['amount']; $data['subject'] = $order['title']; $data['out_trade_no'] = $order['order_no']; }
if ($type == "freeze") { $data['out_order_no'] = $order['order_no']; $data['out_request_no'] = $order['request_no']; $data['order_title'] = $order['title']; $data['amount'] = $order['amount']; }
if ($type == "unfreeze") { $data['auth_no'] = $order['auth_no']; $data['out_request_no'] = $order['request_no']; $data['amount'] = $order['amount']; $data['remark'] = $order['remark']; }
if ($type == "refund"){ $data['trade_no'] = $order['trade_no']; $data['refund_amount'] = $order['refund_amount']; $data['out_request_no'] = $order['out_request_no']; $data['form'] = isset($order['form']) ? $order['form'] : "app"; }
return $data; }
|
