mirror of
https://gitee.com/zoujingli/WeChatDeveloper.git
synced 2025-04-05 19:41:44 +08:00
Merge remote-tracking branch 'origin/master'
This commit is contained in:
commit
4802a7c6a5
@ -83,24 +83,5 @@ class Transfer extends BasicAliPay
|
||||
return $this->getResult($options);
|
||||
}
|
||||
|
||||
/**
|
||||
* 新版 设置网关应用公钥证书SN、支付宝根证书SN
|
||||
*/
|
||||
protected function setAppCertSnAndRootCertSn()
|
||||
{
|
||||
if (!$this->config->get('app_cert')) {
|
||||
throw new InvalidArgumentException("Missing Config -- [app_cert]");
|
||||
}
|
||||
if (!$this->config->get('root_cert')) {
|
||||
throw new InvalidArgumentException("Missing Config -- [root_cert]");
|
||||
}
|
||||
$this->options->set('app_cert_sn', $this->getCertSN($this->config->get('app_cert')));
|
||||
$this->options->set('alipay_root_cert_sn', $this->getRootCertSN($this->config->get('root_cert')));
|
||||
if (!$this->options->get('app_cert_sn')) {
|
||||
throw new InvalidArgumentException("Missing options -- [app_cert_sn]");
|
||||
}
|
||||
if (!$this->options->get('alipay_root_cert_sn')) {
|
||||
throw new InvalidArgumentException("Missing options -- [alipay_root_cert_sn]");
|
||||
}
|
||||
}
|
||||
|
||||
}
|
@ -297,10 +297,9 @@ abstract class BasicAliPay
|
||||
* @param string $sign
|
||||
* @return string
|
||||
*/
|
||||
public function getCertSN($sign)
|
||||
private function getAppCertSN($sign)
|
||||
{
|
||||
// if (file_exists($sign)) $sign = file_get_contents($sign);
|
||||
$ssl = openssl_x509_parse($sign);
|
||||
$ssl = openssl_x509_parse($sign, true);
|
||||
return md5($this->_arr2str(array_reverse($ssl['issuer'])) . $ssl['serialNumber']);
|
||||
}
|
||||
|
||||
@ -309,27 +308,57 @@ abstract class BasicAliPay
|
||||
* @param string $sign
|
||||
* @return string|null
|
||||
*/
|
||||
public function getRootCertSN($sign)
|
||||
private function getRootCertSN($sign)
|
||||
{
|
||||
$sn = null;
|
||||
// if (file_exists($sign)) $sign = file_get_contents($sign);
|
||||
$array = explode("-----END CERTIFICATE-----", $sign);
|
||||
$array = explode('-----END CERTIFICATE-----', $sign);
|
||||
for ($i = 0; $i < count($array) - 1; $i++) {
|
||||
$ssl[$i] = openssl_x509_parse($array[$i] . "-----END CERTIFICATE-----");
|
||||
$ssl[$i] = openssl_x509_parse($array[$i] . '-----END CERTIFICATE-----', true);
|
||||
if (strpos($ssl[$i]['serialNumber'], '0x') === 0) {
|
||||
$ssl[$i]['serialNumber'] = $this->_hex2dec($ssl[$i]['serialNumber']);
|
||||
$ssl[$i]['serialNumber'] = $this->_hex2dec(isset($ssl[$i]['serialNumberHex']) ? $ssl[$i]['serialNumberHex'] : $ssl[$i]['serialNumber']);
|
||||
}
|
||||
if ($ssl[$i]['signatureTypeLN'] == "sha1WithRSAEncryption" || $ssl[$i]['signatureTypeLN'] == "sha256WithRSAEncryption") {
|
||||
if ($sn == null) {
|
||||
$sn = md5($this->_arr2str(array_reverse($ssl[$i]['issuer'])) . $ssl[$i]['serialNumber']);
|
||||
} else {
|
||||
$sn = $sn . "_" . md5($this->_arr2str(array_reverse($ssl[$i]['issuer'])) . $ssl[$i]['serialNumber']);
|
||||
$sn = $sn . '_' . md5($this->_arr2str(array_reverse($ssl[$i]['issuer'])) . $ssl[$i]['serialNumber']);
|
||||
}
|
||||
}
|
||||
}
|
||||
return $sn;
|
||||
}
|
||||
|
||||
/**
|
||||
* 新版 设置网关应用公钥证书SN、支付宝根证书SN
|
||||
*/
|
||||
protected function setAppCertSnAndRootCertSn()
|
||||
{
|
||||
$appCert = $this->config->get('app_cert');
|
||||
$rootCert = $this->config->get('root_cert');
|
||||
$appCertPath = $this->config->get('app_cert_path');
|
||||
$rootCertPath = $this->config->get('root_cert_path');
|
||||
if (empty($appCert) && !empty($appCertPath) && is_file($appCertPath)) {
|
||||
$appCert = file_get_contents($appCertPath);
|
||||
}
|
||||
if (empty($rootCert) && !empty($rootCertPath) && is_file($rootCertPath)) {
|
||||
$rootCert = file_get_contents($rootCertPath);
|
||||
}
|
||||
if (empty($appCert)) {
|
||||
throw new InvalidArgumentException('Missing Config -- [app_cert|app_cert_path]');
|
||||
}
|
||||
if (empty($rootCert)) {
|
||||
throw new InvalidArgumentException('Missing Config -- [root_cert|root_cert_path]');
|
||||
}
|
||||
$this->options->set('app_cert_sn', $this->getAppCertSN($appCert));
|
||||
$this->options->set('alipay_root_cert_sn', $this->getRootCertSN($rootCert));
|
||||
if (!$this->options->get('app_cert_sn')) {
|
||||
throw new InvalidArgumentException('Missing options -- [app_cert_sn]');
|
||||
}
|
||||
if (!$this->options->get('alipay_root_cert_sn')) {
|
||||
throw new InvalidArgumentException('Missing options -- [alipay_root_cert_sn]');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 新版 数组转字符串
|
||||
* @param array $array
|
||||
@ -346,7 +375,6 @@ abstract class BasicAliPay
|
||||
return implode(',', $string);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 新版 0x转高精度数字
|
||||
* @param string $hex
|
||||
@ -367,5 +395,4 @@ abstract class BasicAliPay
|
||||
* @return mixed
|
||||
*/
|
||||
abstract public function apply($options);
|
||||
|
||||
}
|
@ -92,7 +92,7 @@ class BasicPushEvent
|
||||
$this->appid = $this->config->get('appid');
|
||||
// 推送消息处理
|
||||
if ($_SERVER['REQUEST_METHOD'] == "POST") {
|
||||
$this->postxml = file_get_contents("php://input");
|
||||
$this->postxml = Tools::getRawInput();
|
||||
$this->encryptType = $this->input->get('encrypt_type');
|
||||
if ($this->isEncrypt()) {
|
||||
if (empty($options['encodingaeskey'])) {
|
||||
@ -157,16 +157,15 @@ class BasicPushEvent
|
||||
|
||||
/**
|
||||
* 验证来自微信服务器
|
||||
* @param string $str
|
||||
* @return bool
|
||||
*/
|
||||
private function checkSignature($str = '')
|
||||
private function checkSignature()
|
||||
{
|
||||
$nonce = $this->input->get('nonce');
|
||||
$timestamp = $this->input->get('timestamp');
|
||||
$msg_signature = $this->input->get('msg_signature');
|
||||
$signature = empty($msg_signature) ? $this->input->get('signature') : $msg_signature;
|
||||
$tmpArr = [$this->config->get('token'), $timestamp, $nonce, $str];
|
||||
$tmpArr = [$this->config->get('token'), $timestamp, $nonce, ''];
|
||||
sort($tmpArr, SORT_STRING);
|
||||
return sha1(implode($tmpArr)) === $signature;
|
||||
}
|
||||
|
@ -92,12 +92,13 @@ class BasicWePay
|
||||
|
||||
/**
|
||||
* 获取微信支付通知
|
||||
* @param string $xml
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
*/
|
||||
public function getNotify()
|
||||
public function getNotify($xml = '')
|
||||
{
|
||||
$data = Tools::xml2arr(file_get_contents('php://input'));
|
||||
$data = Tools::xml2arr(empty($xml) ? Tools::getRawInput() : $xml);
|
||||
if (isset($data['sign']) && $this->getPaySign($data) === $data['sign']) {
|
||||
return $data;
|
||||
}
|
||||
|
@ -53,7 +53,7 @@ class DataArray implements ArrayAccess
|
||||
/**
|
||||
* 获取配置项参数
|
||||
* @param string|null $offset
|
||||
* @return array|string|null
|
||||
* @return array|string|null|mixed
|
||||
*/
|
||||
public function get($offset = null)
|
||||
{
|
||||
@ -117,7 +117,7 @@ class DataArray implements ArrayAccess
|
||||
/**
|
||||
* 获取配置项参数
|
||||
* @param string|null $offset
|
||||
* @return array|string|null
|
||||
* @return array|string|null|mixed
|
||||
*/
|
||||
#[\ReturnTypeWillChange]
|
||||
public function offsetGet($offset = null)
|
||||
|
@ -65,6 +65,18 @@ class Tools
|
||||
return $str;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取输入对象
|
||||
* @return false|mixed|string
|
||||
*/
|
||||
public static function getRawInput()
|
||||
{
|
||||
if (empty($GLOBALS['HTTP_RAW_POST_DATA'])) {
|
||||
return file_get_contents('php://input');
|
||||
} else {
|
||||
return $GLOBALS['HTTP_RAW_POST_DATA'];
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 根据文件后缀获取文件类型
|
||||
|
@ -57,13 +57,14 @@ class Refund extends BasicWePay
|
||||
|
||||
/**
|
||||
* 获取退款通知
|
||||
* @param string $xml
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidDecryptException
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
*/
|
||||
public function getNotify()
|
||||
public function getNotify($xml = '')
|
||||
{
|
||||
$data = Tools::xml2arr(file_get_contents("php://input"));
|
||||
$data = Tools::xml2arr(empty($xml) ? Tools::getRawInput() : $xml);
|
||||
if (!isset($data['return_code']) || $data['return_code'] !== 'SUCCESS') {
|
||||
throw new InvalidResponseException('获取退款通知XML失败!');
|
||||
}
|
||||
|
@ -18,6 +18,7 @@ namespace WePayV3\Contracts;
|
||||
|
||||
use WeChat\Contracts\Tools;
|
||||
use WeChat\Exceptions\InvalidArgumentException;
|
||||
use WeChat\Exceptions\InvalidDecryptException;
|
||||
use WeChat\Exceptions\InvalidResponseException;
|
||||
use WePayV3\Cert;
|
||||
|
||||
@ -65,12 +66,12 @@ abstract class BasicWePay
|
||||
if (empty($options['mch_v3_key'])) {
|
||||
throw new InvalidArgumentException("Missing Config -- [mch_v3_key]");
|
||||
}
|
||||
if (empty($options['cert_private'])) {
|
||||
throw new InvalidArgumentException("Missing Config -- [cert_private]");
|
||||
}
|
||||
if (empty($options['cert_public'])) {
|
||||
throw new InvalidArgumentException("Missing Config -- [cert_public]");
|
||||
}
|
||||
if (empty($options['cert_private'])) {
|
||||
throw new InvalidArgumentException("Missing Config -- [cert_private]");
|
||||
}
|
||||
|
||||
if (stripos($options['cert_public'], '-----BEGIN CERTIFICATE-----') === false) {
|
||||
if (file_exists($options['cert_public'])) {
|
||||
@ -93,11 +94,33 @@ abstract class BasicWePay
|
||||
$this->config['mch_v3_key'] = $options['mch_v3_key'];
|
||||
$this->config['cert_public'] = $options['cert_public'];
|
||||
$this->config['cert_private'] = $options['cert_private'];
|
||||
$this->config['cert_serial'] = openssl_x509_parse($this->config['cert_public'])['serialNumberHex'];
|
||||
|
||||
if (empty($this->config['cert_serial'])) {
|
||||
throw new InvalidArgumentException("Failed to parse certificate public key");
|
||||
if (empty($options['cert_serial'])) {
|
||||
$this->config['cert_serial'] = openssl_x509_parse($this->config['cert_public'], true)['serialNumberHex'];
|
||||
} else {
|
||||
$this->config['cert_serial'] = $options['cert_serial'];
|
||||
}
|
||||
if (empty($this->config['cert_serial'])) {
|
||||
throw new InvalidArgumentException('Failed to parse certificate public key');
|
||||
}
|
||||
|
||||
if (!empty($options['cache_path'])) {
|
||||
Tools::$cache_path = $options['cache_path'];
|
||||
}
|
||||
|
||||
// 服务商参数支持
|
||||
// if (!empty($options['sp_appid'])) {
|
||||
// $this->config['sp_appid'] = $options['sp_appid'];
|
||||
// }
|
||||
// if (!empty($options['sp_mchid'])) {
|
||||
// $this->config['sp_mchid'] = $options['sp_mchid'];
|
||||
// }
|
||||
// if (!empty($options['sub_appid'])) {
|
||||
// $this->config['sub_appid'] = $options['sub_appid'];
|
||||
// }
|
||||
// if (!empty($options['sub_mch_id'])) {
|
||||
// $this->config['sub_mch_id'] = $options['sub_mch_id'];
|
||||
// }
|
||||
}
|
||||
|
||||
/**
|
||||
@ -117,24 +140,31 @@ abstract class BasicWePay
|
||||
* @param string $method 请求访问
|
||||
* @param string $pathinfo 请求路由
|
||||
* @param string $jsondata 请求数据
|
||||
* @param bool $verify 是否验证
|
||||
* @return array
|
||||
* @param boolean $verify 是否验证
|
||||
* @param boolean $isjson 返回JSON
|
||||
* @return array|string
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
*/
|
||||
public function doRequest($method, $pathinfo, $jsondata = '', $verify = false)
|
||||
public function doRequest($method, $pathinfo, $jsondata = '', $verify = false, $isjson = true)
|
||||
{
|
||||
list($time, $nonce) = [time(), uniqid() . rand(1000, 9999)];
|
||||
$signstr = join("\n", [$method, $pathinfo, $time, $nonce, $jsondata, '']);
|
||||
|
||||
// 生成数据签名TOKEN
|
||||
$token = sprintf('mchid="%s",nonce_str="%s",timestamp="%d",serial_no="%s",signature="%s"',
|
||||
$this->config['mch_id'], $nonce, $time, $this->config['cert_serial'], $this->signBuild($signstr)
|
||||
);
|
||||
list($header, $content) = $this->_doRequestCurl($method, $this->base . $pathinfo, [
|
||||
$location = (preg_match('|^https?://|', $pathinfo) ? '' : $this->base) . $pathinfo;
|
||||
list($header, $content) = $this->_doRequestCurl($method, $location, [
|
||||
'data' => $jsondata, 'header' => [
|
||||
"Accept: application/json", "Content-Type: application/json",
|
||||
'User-Agent: https://thinkadmin.top', "Authorization: WECHATPAY2-SHA256-RSA2048 {$token}",
|
||||
'Accept: application/json',
|
||||
'Content-Type: application/json',
|
||||
'User-Agent: https://thinkadmin.top',
|
||||
"Authorization: WECHATPAY2-SHA256-RSA2048 {$token}",
|
||||
"Wechatpay-Serial: {$this->config['cert_serial']}"
|
||||
],
|
||||
]);
|
||||
|
||||
if ($verify) {
|
||||
if(empty($headers)){
|
||||
return json_decode($content, true);
|
||||
@ -148,15 +178,19 @@ abstract class BasicWePay
|
||||
}
|
||||
}
|
||||
try {
|
||||
if (empty($headers)) {
|
||||
return $isjson ? json_decode($content, true) : $content;
|
||||
}
|
||||
$string = join("\n", [$headers['timestamp'], $headers['nonce'], $content, '']);
|
||||
if (!$this->signVerify($string, $headers['signature'], $headers['serial'])) {
|
||||
throw new InvalidResponseException("验证响应签名失败");
|
||||
throw new InvalidResponseException('验证响应签名失败');
|
||||
}
|
||||
} catch (\Exception $exception) {
|
||||
throw new InvalidResponseException($exception->getMessage(), $exception->getCode());
|
||||
}
|
||||
}
|
||||
return json_decode($content, true);
|
||||
|
||||
return $isjson ? json_decode($content, true) : $content;
|
||||
}
|
||||
|
||||
/**
|
||||
@ -236,4 +270,36 @@ abstract class BasicWePay
|
||||
return Tools::setCache($name, base64_encode($content), 7200);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* RSA加密处理
|
||||
* @param string $string
|
||||
* @return string
|
||||
* @throws \WeChat\Exceptions\InvalidDecryptException
|
||||
*/
|
||||
protected function rsaEncode($string)
|
||||
{
|
||||
$publicKey = file_get_contents($this->config['cert_public']);
|
||||
if (openssl_public_encrypt($string, $encrypted, $publicKey, OPENSSL_PKCS1_OAEP_PADDING)) {
|
||||
return base64_encode($encrypted);
|
||||
} else {
|
||||
throw new InvalidDecryptException('Rsa Encrypt Error.');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* RSA 解密处理
|
||||
* @param string $string
|
||||
* @return string
|
||||
* @throws \WeChat\Exceptions\InvalidDecryptException
|
||||
*/
|
||||
protected function rsaDecode($string)
|
||||
{
|
||||
$private = file_get_contents($this->config['cert_private']);
|
||||
if (openssl_private_decrypt(base64_decode($string), $content, $private, OPENSSL_PKCS1_OAEP_PADDING)) {
|
||||
return $content;
|
||||
} else {
|
||||
throw new InvalidDecryptException('Rsa Decrypt Error.');
|
||||
}
|
||||
}
|
||||
}
|
@ -18,11 +18,13 @@ namespace WePayV3;
|
||||
|
||||
use WeChat\Contracts\Tools;
|
||||
use WeChat\Exceptions\InvalidArgumentException;
|
||||
use WeChat\Exceptions\InvalidDecryptException;
|
||||
use WeChat\Exceptions\InvalidResponseException;
|
||||
use WePayV3\Contracts\BasicWePay;
|
||||
use WePayV3\Contracts\DecryptAes;
|
||||
|
||||
/**
|
||||
* 订单支付接口
|
||||
* 直连商户 | 订单支付接口
|
||||
* Class Order
|
||||
* @package WePayV3
|
||||
*/
|
||||
@ -39,6 +41,7 @@ class Order extends BasicWePay
|
||||
* @param array $data 支付参数
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @document https://pay.weixin.qq.com/wiki/doc/apiv3/apis/chapter3_1_1.shtml
|
||||
*/
|
||||
public function create($type, $data)
|
||||
{
|
||||
@ -53,9 +56,13 @@ class Order extends BasicWePay
|
||||
} else {
|
||||
// 创建预支付码
|
||||
$result = $this->doRequest('POST', $types[$type], json_encode($data, JSON_UNESCAPED_UNICODE), true);
|
||||
if (empty($result['prepay_id'])) return $result;
|
||||
if (empty($result['h5_url']) && empty($result['code_url']) && empty($result['prepay_id'])) {
|
||||
$message = isset($result['code']) ? "[ {$result['code']} ] " : '';
|
||||
$message .= isset($result['message']) ? $result['message'] : json_encode($result, JSON_UNESCAPED_UNICODE);
|
||||
throw new InvalidResponseException($message);
|
||||
}
|
||||
// 支付参数签名
|
||||
$time = (string)time();
|
||||
$time = strval(time());
|
||||
$appid = $this->config['appid'];
|
||||
$prepayId = $result['prepay_id'];
|
||||
$nonceStr = Tools::createNoncestr();
|
||||
@ -73,30 +80,42 @@ class Order extends BasicWePay
|
||||
|
||||
/**
|
||||
* 支付订单查询
|
||||
* @param string $orderNo 订单单号
|
||||
* @param string $tradeNo 订单单号
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @document https://pay.weixin.qq.com/wiki/doc/apiv3/apis/chapter3_1_2.shtml
|
||||
*/
|
||||
public function query($orderNo)
|
||||
public function query($tradeNo)
|
||||
{
|
||||
$pathinfo = "/v3/pay/transactions/out-trade-no/{$orderNo}";
|
||||
$pathinfo = "/v3/pay/transactions/out-trade-no/{$tradeNo}";
|
||||
return $this->doRequest('GET', "{$pathinfo}?mchid={$this->config['mch_id']}", '', true);
|
||||
}
|
||||
|
||||
/**
|
||||
* 支付通知
|
||||
* 关闭支付订单
|
||||
* @param string $tradeNo 订单单号
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
*/
|
||||
public function close($tradeNo)
|
||||
{
|
||||
$data = ['mchid' => $this->config['mch_id']];
|
||||
$path = "/v3/pay/transactions/out-trade-no/{$tradeNo}/close";
|
||||
return $this->doRequest('POST', $path, json_encode($data, JSON_UNESCAPED_UNICODE), true);
|
||||
}
|
||||
|
||||
/**
|
||||
* 支付通知解析
|
||||
* @param array $data
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidDecryptException
|
||||
*/
|
||||
public function notify(array $parameters = [])
|
||||
public function notify(array $data = [])
|
||||
{
|
||||
if (empty($parameters)) {
|
||||
$body = file_get_contents('php://input');
|
||||
if (empty($data)) {
|
||||
$body = Tools::getRawInput();
|
||||
$data = json_decode($body, true);
|
||||
} else {
|
||||
$data = $parameters;
|
||||
}
|
||||
|
||||
if (isset($data['resource'])) {
|
||||
$aes = new DecryptAes($this->config['mch_v3_key']);
|
||||
$data['result'] = $aes->decryptToString(
|
||||
@ -107,4 +126,92 @@ class Order extends BasicWePay
|
||||
}
|
||||
return $data;
|
||||
}
|
||||
|
||||
/**
|
||||
* 创建退款订单
|
||||
* @param array $data 退款参数
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @document https://pay.weixin.qq.com/wiki/doc/apiv3/apis/chapter3_1_9.shtml
|
||||
*/
|
||||
public function createRefund($data)
|
||||
{
|
||||
$path = '/v3/refund/domestic/refunds';
|
||||
return $this->doRequest('POST', $path, json_encode($data, JSON_UNESCAPED_UNICODE), true);
|
||||
}
|
||||
|
||||
/**
|
||||
* 退款订单查询
|
||||
* @param string $refundNo 退款单号
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @document https://pay.weixin.qq.com/wiki/doc/apiv3/apis/chapter3_1_10.shtml
|
||||
*/
|
||||
public function queryRefund($refundNo)
|
||||
{
|
||||
$path = "/v3/refund/domestic/refunds/{$refundNo}";
|
||||
return $this->doRequest('GET', $path, '', true);
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取退款通知
|
||||
* @param string $xml
|
||||
* @return array
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidDecryptException
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
*/
|
||||
public function notifyRefund($xml = '')
|
||||
{
|
||||
$data = Tools::xml2arr(empty($xml) ? Tools::getRawInput() : $xml);
|
||||
if (empty($data['return_code']) || $data['return_code'] !== 'SUCCESS') {
|
||||
throw new InvalidResponseException('获取退款通知失败!');
|
||||
}
|
||||
try {
|
||||
$decrypt = base64_decode($data['req_info']);
|
||||
$response = openssl_decrypt($decrypt, 'aes-256-ecb', md5($this->config['mch_v3_key']), OPENSSL_RAW_DATA);
|
||||
$data['result'] = Tools::xml2arr($response);
|
||||
return $data;
|
||||
} catch (\Exception $exception) {
|
||||
throw new InvalidDecryptException($exception->getMessage(), $exception->getCode());
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 申请交易账单
|
||||
* @param array|string $params
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @document https://pay.weixin.qq.com/wiki/doc/apiv3/apis/chapter3_3_6.shtml
|
||||
*/
|
||||
public function tradeBill($params)
|
||||
{
|
||||
$path = '/v3/bill/tradebill?' . is_array($params) ? http_build_query($params) : $params;
|
||||
return $this->doRequest('GET', $path, '', true);
|
||||
}
|
||||
|
||||
/**
|
||||
* 申请资金账单
|
||||
* @param array|string $params
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @document https://pay.weixin.qq.com/wiki/doc/apiv3/apis/chapter3_3_7.shtml
|
||||
*/
|
||||
public function fundflowBill($params)
|
||||
{
|
||||
$path = '/v3/bill/fundflowbill?' . is_array($params) ? http_build_query($params) : $params;
|
||||
return $this->doRequest('GET', $path, '', true);
|
||||
}
|
||||
|
||||
/**
|
||||
* 下载账单文件
|
||||
* @param string $fileurl
|
||||
* @return string 二进制 Excel 内容
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @document https://pay.weixin.qq.com/wiki/doc/apiv3_partner/apis/chapter7_6_1.shtml
|
||||
*/
|
||||
public function downloadBill($fileurl)
|
||||
{
|
||||
return $this->doRequest('GET', $fileurl, '', false, false);
|
||||
}
|
||||
}
|
||||
|
@ -25,8 +25,6 @@ use WePayV3\Contracts\BasicWePay;
|
||||
*/
|
||||
class ProfitSharing extends BasicWePay
|
||||
{
|
||||
|
||||
|
||||
/**
|
||||
* 请求分账
|
||||
* @param array $options
|
||||
@ -63,9 +61,10 @@ class ProfitSharing extends BasicWePay
|
||||
{
|
||||
return $this->doRequest('POST', '/v3/profitsharing/orders/unfreeze', json_encode($options, JSON_UNESCAPED_UNICODE), true);
|
||||
}
|
||||
|
||||
/**
|
||||
* 查询剩余待分金额
|
||||
* @param array $transactionId 微信订单号
|
||||
* @param string $transactionId 微信订单号
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
*/
|
||||
@ -74,6 +73,7 @@ class ProfitSharing extends BasicWePay
|
||||
$pathinfo = "/v3/profitsharing/transactions/{$transactionId}/amounts";
|
||||
return $this->doRequest('GET', $pathinfo, '', true);
|
||||
}
|
||||
|
||||
/**
|
||||
* 添加分账接收方
|
||||
* @param array $options
|
||||
@ -85,6 +85,7 @@ class ProfitSharing extends BasicWePay
|
||||
$options['appid'] = $this->config['appid'];
|
||||
return $this->doRequest('POST', "/v3/profitsharing/receivers/add", json_encode($options, JSON_UNESCAPED_UNICODE), true);
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除分账接收方
|
||||
* @param array $options
|
||||
|
@ -22,8 +22,10 @@ use WeChat\Exceptions\InvalidResponseException;
|
||||
use WePayV3\Contracts\BasicWePay;
|
||||
|
||||
/**
|
||||
* 订单退款接口
|
||||
* Class Refund
|
||||
* 电商接口 | 订单退款接口
|
||||
* 注意:直连商户退款接口集成在 Order 中
|
||||
* @deprecated
|
||||
* @class Refund
|
||||
* @package WePayV3
|
||||
*/
|
||||
class Refund extends BasicWePay
|
||||
@ -53,13 +55,15 @@ class Refund extends BasicWePay
|
||||
|
||||
/**
|
||||
* 获取退款通知
|
||||
* @param string $xml
|
||||
* @return array
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidDecryptException
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
*/
|
||||
public function notify()
|
||||
public function notify($xml = '')
|
||||
{
|
||||
$data = Tools::xml2arr(file_get_contents("php://input"));
|
||||
$data = Tools::xml2arr(empty($xml) ? Tools::getRawInput() : $xml);
|
||||
if (!isset($data['return_code']) || $data['return_code'] !== 'SUCCESS') {
|
||||
throw new InvalidResponseException('获取退款通知XML失败!');
|
||||
}
|
||||
|
@ -20,7 +20,7 @@ use WePayV3\Contracts\BasicWePay;
|
||||
|
||||
/**
|
||||
* 普通商户商家转账到零钱
|
||||
* Class Transfers
|
||||
* @class Transfers
|
||||
* @package WePayV3
|
||||
*/
|
||||
class Transfers extends BasicWePay
|
||||
@ -29,10 +29,16 @@ class Transfers extends BasicWePay
|
||||
* 发起商家批量转账
|
||||
* @param array $body
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidDecryptException
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
*/
|
||||
public function batchs($body)
|
||||
{
|
||||
if (isset($body['transfer_detail_list']) && is_array($body['transfer_detail_list'])) {
|
||||
foreach ($body['transfer_detail_list'] as &$item) if (isset($item['user_name'])) {
|
||||
$item['user_name'] = $this->rsaEncode($item['user_name']);
|
||||
}
|
||||
}
|
||||
return $this->doRequest('POST', '/v3/transfer/batches', json_encode($body, JSON_UNESCAPED_UNICODE), true);
|
||||
}
|
||||
|
||||
|
@ -28,7 +28,7 @@ try {
|
||||
|
||||
// 请参考(请求参数):https://docs.open.alipay.com/api_1/alipay.trade.app.pay
|
||||
$result = $pay->apply([
|
||||
'out_trade_no' => time(), // 商户订单号
|
||||
'out_trade_no' => strval(time()), // 商户订单号
|
||||
'total_amount' => '1', // 支付金额
|
||||
'subject' => '支付宝订单标题', // 支付订单描述
|
||||
]);
|
||||
|
@ -14,21 +14,41 @@
|
||||
// | github 代码仓库:https://github.com/zoujingli/WeChatDeveloper
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
/**
|
||||
* 【 名词解释 】
|
||||
* 应用私钥:用来给应用消息进行签名,请务必要妥善保管,避免遗失或泄露。
|
||||
* 应用公钥:需要提供给支付宝开放平台,平台会对应用发送的消息进行签名验证。
|
||||
* 支付宝公钥:应用收到支付宝发送的同步、异步消息时,使用支付宝公钥验证签名信息。
|
||||
* CSR 文件:CSR 即证书签名请求(Certificate Signing Request),CSR 文件(.csr)是申请证书时所需要的一个数据文件。
|
||||
* 应用公钥证书:在开放平台上传 CSR 文件后可以获取 CA 机构颁发的应用证书文件(.crt),其中包含了组织/公司名称、应用公钥、证书有效期等内容,一般有效期为 5 年。
|
||||
* 支付宝公钥证书:用来验证支付宝消息,包含了支付宝公钥、支付宝公司名称、证书有效期等内容,一般有效期为 5 年。
|
||||
* 支付宝根证书:用来验证支付宝消息,包含了根 CA 名称、根 CA 的公钥、证书有效期等内容。
|
||||
*/
|
||||
|
||||
/**
|
||||
* 应用公钥证书SN(app_cert_sn)和支付宝根证书SN(alipay_root_cert_sn)的 sn 是指什么?
|
||||
* 使用公钥证书签名方式下, 请求参数中需要携带应用公钥证书SN(app_cert_sn)、支付宝根证书SN(alipay_root_cert_sn),这里的SN是指基于开放平台提供的计算规则,动态计算出来的公钥证书序列号,与X.509证书中内置的序列号(serialNumber)不同。
|
||||
* 具体的计算规则如下:解析X.509证书文件,获取证书签发机构名称(name)以及证书内置序列号(serialNumber)。 将name与serialNumber拼接成字符串,再对该字符串做MD5计算。
|
||||
* 可以参考开放平台SDK源码中的 AlipaySignature.getCertSN 方法实现
|
||||
*
|
||||
* 不直接使用证书文件中内置的序列号原因:开放平台支持开发者上传自己找第三方权威CA签发的证书,而证书文件中内置序列号只能保证同一家签发机构签发的证书不重复
|
||||
*/
|
||||
|
||||
return [
|
||||
// 沙箱模式
|
||||
'debug' => true,
|
||||
// 签名类型(RSA|RSA2)
|
||||
'sign_type' => "RSA2",
|
||||
'sign_type' => 'RSA2',
|
||||
// 应用ID
|
||||
'appid' => '2016090900468879',
|
||||
// 应用私钥的内容 (1行填写,特别注意:这里的应用私钥通常由支付宝密钥管理工具生成)
|
||||
'private_key' => 'MIIEpAIBAAKCAQEArr8ymPBh/lQ46y76f3VZW4HGV8tClTHJhhlt7BPmn0F8fSc3alKFCiHXIT01ccy3xTPVgcYX26/vvQBt9cJZqxEWapj4hb8BH6lr5BH5MMQ88eyyPLlYicGzt5S4iXUjuImsUguaCO12SoFmM1eaMYKqrELkg3Apbc+16Ktq+puKAohZeozd80NnDc1ossrZtHU1DkUkEJrBeaC7/D+0H/HBHJzc60EMYpvSGgyKzP/ke6xF4ZGilFSKsdoKgS8u7paVMh2SlSO5AsMoELTTlsFS4eYOgD6ZThkBTZZIcyKHM1Wjq6qDfhU3oTkJDYqnLWdtzoqgnUPNQ8nYpYrdeQIDAQABAoIBAQCKWRmH+Bi9MJT3rfPo4VFjnzUW4PfQAuDX6F4coAzgXQpgU6IN7VMjGHOn/zvG4xtDZ6xL2DefWIVnj2V/QuWXCCpFLuLjkLslBA9FO+2b7GGL76eVZ/Bu8AqG95m6SiGDwovJUSIcm1Qh3Jy7XUnYlOjnBPbCERTbuaz9jmleCmOmh4RnpL0DmzvUffNmEuqJfgnF0h2h6CwYBUyGY8G3zpGAyqY+viVPjKA8catcQTqDRywe3ktXxIbi0JqTAYDpb81Ih4NQeHxGMUWHabwij9UTYdPYBgzwVxjcJF4O1cPGdnYeRXm8Neq6L2HbEPRqe/Jiw4i4AB4V3cVWExPhAoGBANWyjP43kguThkFsQZPfDP5xuXIQJ5W24ZcDIwcFYZ5glh+hddHuWsrCf68VHXsf2epeNdqz57xbdtg84FI8N0HmjTCtfuagxXUMPbX4NqqzPqOu2m0T5/b8HCEDV3zWqfMOt+kHR4NytFxLKLwR0a9kypsfZ2z/X4BYXBi6PiZFAoGBANFWw+AL9hJ2q51ePTcZrE788hPepcsdv3OfJjnspykKighTiFgT+m+Kr/AEa9IU/njz/+tcWJuAwOPKDFVd0zVRcxK2Y5WLfo8fYEt6yhq8oa5OKFwInXXVmCPMn0+qduNmZMTzSqL2kaN0U+AH5Te1vlv7I7yuwHDE9vwknBelAoGAbpPN0V3//G2B8yiJZnLszl0akKM7WIUhhnrhDSkDsmhYRlXOGas03+Z1G6vZbXS11kiZpWmiaB0MCii2CteN4FPki2O7XquigUasSBUAdKP7rcc0z2yVg4BBLfQEuVx65IKhN7vEjYg1O+zIT0kJL7EABfTiF8ytJkSSo1j7/+ECgYEA0NPuKHWmLvsE7cKR7IKG2nEIiHvGBl6RmyS7PHNwucdStUWnML4VSOof4p52dKcOx9gYh1Ci79U8FsB7Fzm2tWygD520r/zs7peNNx6xuIROAZTkPBM4CNFfqO66Sf2yBd0iTzqoTPMNi/JCra0So0WBNT7Ngq8NODG0dQmMUSUCgYANC6Y4JIq24nh27jZjTlsVA+jkv5tFK2aKnfRwOYcNhjVNaMyMz5BbvE07zlr/CgViauEMEJorTs+eNY1TEW7ZOOr2lFgbUnT+h/GLCvgwV00nYBEimfIJ96jfIotOLYB74oJJni9wCaJCuurLNfJ0E1XS/8d+1BkFQ+Q7VSDYpg==',
|
||||
// 支付宝公钥内容 (1行填写,特别注意:这里不是应用公钥而是支付宝公钥,通常是上传应用公钥换取支付宝公钥,在网页可以复制)
|
||||
// 支付宝公钥内容 (需1行填写,特别注意:这里不是应用公钥而是支付宝公钥,通常是上传应用公钥换取支付宝公钥,在网页可以复制)
|
||||
'public_key' => 'MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAtU71NY53UDGY7JNvLYAhsNa+taTF6KthIHJmGgdio9bkqeJGhHk6ttkTKkLqFgwIfgAkHpdKiOv1uZw6gVGZ7TCu5LfHTqKrCd6Uz+N7hxhY+4IwicLgprcV1flXQLmbkJYzFMZqkXGkSgOsR2yXh4LyQZczgk9N456uuzGtRy7MoB4zQy34PLUkkxR6W1B2ftNbLRGXv6tc7p/cmDcrY6K1bSxnGmfRxFSb8lRfhe0V0UM6pKq2SGGSeovrKHN0OLp+Nn5wcULVnFgATXGCENshRlp96piPEBFwneXs19n+sX1jx60FTR7/rME3sW3AHug0fhZ9mSqW4x401WjdnwIDAQAB',
|
||||
// 应用公钥的内容(新版资金类接口转 app_cert_sn)
|
||||
'app_cert' => '',
|
||||
// 支付宝根证书内容(新版资金类接口转 alipay_root_cert_sn)
|
||||
'root_cert' => '',
|
||||
// 应用私钥的内容 (需1行填写,特别注意:这里的应用私钥通常由支付宝密钥管理工具生成)
|
||||
'private_key' => 'MIIEpAIBAAKCAQEArr8ymPBh/lQ46y76f3VZW4HGV8tClTHJhhlt7BPmn0F8fSc3alKFCiHXIT01ccy3xTPVgcYX26/vvQBt9cJZqxEWapj4hb8BH6lr5BH5MMQ88eyyPLlYicGzt5S4iXUjuImsUguaCO12SoFmM1eaMYKqrELkg3Apbc+16Ktq+puKAohZeozd80NnDc1ossrZtHU1DkUkEJrBeaC7/D+0H/HBHJzc60EMYpvSGgyKzP/ke6xF4ZGilFSKsdoKgS8u7paVMh2SlSO5AsMoELTTlsFS4eYOgD6ZThkBTZZIcyKHM1Wjq6qDfhU3oTkJDYqnLWdtzoqgnUPNQ8nYpYrdeQIDAQABAoIBAQCKWRmH+Bi9MJT3rfPo4VFjnzUW4PfQAuDX6F4coAzgXQpgU6IN7VMjGHOn/zvG4xtDZ6xL2DefWIVnj2V/QuWXCCpFLuLjkLslBA9FO+2b7GGL76eVZ/Bu8AqG95m6SiGDwovJUSIcm1Qh3Jy7XUnYlOjnBPbCERTbuaz9jmleCmOmh4RnpL0DmzvUffNmEuqJfgnF0h2h6CwYBUyGY8G3zpGAyqY+viVPjKA8catcQTqDRywe3ktXxIbi0JqTAYDpb81Ih4NQeHxGMUWHabwij9UTYdPYBgzwVxjcJF4O1cPGdnYeRXm8Neq6L2HbEPRqe/Jiw4i4AB4V3cVWExPhAoGBANWyjP43kguThkFsQZPfDP5xuXIQJ5W24ZcDIwcFYZ5glh+hddHuWsrCf68VHXsf2epeNdqz57xbdtg84FI8N0HmjTCtfuagxXUMPbX4NqqzPqOu2m0T5/b8HCEDV3zWqfMOt+kHR4NytFxLKLwR0a9kypsfZ2z/X4BYXBi6PiZFAoGBANFWw+AL9hJ2q51ePTcZrE788hPepcsdv3OfJjnspykKighTiFgT+m+Kr/AEa9IU/njz/+tcWJuAwOPKDFVd0zVRcxK2Y5WLfo8fYEt6yhq8oa5OKFwInXXVmCPMn0+qduNmZMTzSqL2kaN0U+AH5Te1vlv7I7yuwHDE9vwknBelAoGAbpPN0V3//G2B8yiJZnLszl0akKM7WIUhhnrhDSkDsmhYRlXOGas03+Z1G6vZbXS11kiZpWmiaB0MCii2CteN4FPki2O7XquigUasSBUAdKP7rcc0z2yVg4BBLfQEuVx65IKhN7vEjYg1O+zIT0kJL7EABfTiF8ytJkSSo1j7/+ECgYEA0NPuKHWmLvsE7cKR7IKG2nEIiHvGBl6RmyS7PHNwucdStUWnML4VSOof4p52dKcOx9gYh1Ci79U8FsB7Fzm2tWygD520r/zs7peNNx6xuIROAZTkPBM4CNFfqO66Sf2yBd0iTzqoTPMNi/JCra0So0WBNT7Ngq8NODG0dQmMUSUCgYANC6Y4JIq24nh27jZjTlsVA+jkv5tFK2aKnfRwOYcNhjVNaMyMz5BbvE07zlr/CgViauEMEJorTs+eNY1TEW7ZOOr2lFgbUnT+h/GLCvgwV00nYBEimfIJ96jfIotOLYB74oJJni9wCaJCuurLNfJ0E1XS/8d+1BkFQ+Q7VSDYpg==',
|
||||
// 应用公钥的内容(新版资金类接口转 app_cert_sn,如文件 appCertPublicKey_2019051064521003.crt)
|
||||
'app_cert' => '', // 'app_cert_path' => '',
|
||||
// 支付宝根证书的内容(新版资金类接口转 alipay_root_cert_sn,如文件 alipayRootCert.crt)
|
||||
'root_cert' => '', // 'root_cert_path' => ''
|
||||
// 支付成功通知地址
|
||||
'notify_url' => '',
|
||||
// 网页支付回跳地址
|
||||
|
@ -27,14 +27,18 @@ $certPrivate = <<<CERT
|
||||
CERT;
|
||||
|
||||
return [
|
||||
// 商户绑定的公众号APPID
|
||||
// 可选,公众号APPID
|
||||
'appid' => '',
|
||||
// 微信商户编号ID
|
||||
// 必填,微信商户编号ID
|
||||
'mch_id' => '',
|
||||
// 微信商户V3接口密钥
|
||||
// 必填,微信商户V3接口密钥
|
||||
'mch_v3_key' => '',
|
||||
// 微信商户证书公钥,支持证书内容或文件路径
|
||||
// 可选,微信商户证书序列号,可从公钥中提取
|
||||
'cert_serial' => '',
|
||||
// 必填,微信商户证书公钥,支持证书内容或文件路径
|
||||
'cert_public' => $certPublic,
|
||||
// 微信商户证书私钥,支持证书内容或文件路径
|
||||
// 必填,微信商户证书私钥,支持证书内容或文件路径
|
||||
'cert_private' => $certPrivate,
|
||||
// 可选,运行时的文件缓存路径
|
||||
'cache_path' => ''
|
||||
];
|
24
readme.md
24
readme.md
@ -199,19 +199,19 @@ try {
|
||||
```php
|
||||
$config = [
|
||||
// 沙箱模式
|
||||
'debug' => true,
|
||||
'debug' => true,
|
||||
// 签名类型(RSA|RSA2)
|
||||
'sign_type' => "RSA2",
|
||||
// 应用ID
|
||||
'appid' => '2016090900468879',
|
||||
// 支付宝公钥文字内容 (1行填写,特别注意:这里是支付宝公钥,不是应用公钥,最好从开发者中心的网页上去复制)
|
||||
'public_key' => 'MIIBIjANBgkqhkiG9...',
|
||||
// 支付宝私钥文字内容 (1行填写)
|
||||
'private_key' => 'MIIEvQIBADANBgkqh...',
|
||||
// 应用公钥证书完整内容(新版资金类接口转 app_cert_sn)
|
||||
'app_cert' => '',
|
||||
// 支付宝根证书完整内容(新版资金类接口转 alipay_root_cert_sn)
|
||||
'root_cert' => '',
|
||||
'sign_type' => "RSA2",
|
||||
// 应用APPID
|
||||
'appid' => '2016090900468879',
|
||||
// 支付宝公钥内容 (需1行填写,特别注意:这里不是应用公钥而是支付宝公钥,通常是上传应用公钥换取支付宝公钥,在网页可以复制)
|
||||
'public_key' => 'MIIBIjAN...',
|
||||
// 应用私钥的内容 (需1行填写,特别注意:这里的应用私钥通常由支付宝密钥管理工具生成)
|
||||
'private_key' => 'MIIEpAIB...',
|
||||
// 应用公钥的内容(新版资金类接口转 app_cert_sn,如文件 appCertPublicKey_2019051064521003.crt)
|
||||
'app_cert' => '', // 'app_cert_path' => '',
|
||||
// 支付宝根证书的内容(新版资金类接口转 alipay_root_cert_sn,如文件 alipayRootCert.crt)
|
||||
'root_cert' => '', // 'root_cert_path' => ''
|
||||
// 支付成功通知地址
|
||||
'notify_url' => '',
|
||||
// 网页支付回跳地址
|
||||
|
Loading…
x
Reference in New Issue
Block a user