[更新]增加获取支付通知的方法 notify #20 【待测试】

This commit is contained in:
邹景立 2018-11-19 21:24:17 +08:00
parent 7e9ec275be
commit 37c8ab0afe
4 changed files with 82 additions and 49 deletions

View File

@ -15,6 +15,7 @@
namespace WeChat\Contracts; namespace WeChat\Contracts;
use WeChat\Exceptions\InvalidArgumentException; use WeChat\Exceptions\InvalidArgumentException;
use WeChat\Exceptions\InvalidResponseException;
/** /**
* 支付宝支付基类 * 支付宝支付基类
@ -91,7 +92,7 @@ abstract class BasicAliPay
* 查询支付宝订单状态 * 查询支付宝订单状态
* @param string $out_trade_no * @param string $out_trade_no
* @return array|boolean * @return array|boolean
* @throws \WeChat\Exceptions\InvalidResponseException * @throws InvalidResponseException
*/ */
public function query($out_trade_no = '') public function query($out_trade_no = '')
{ {
@ -104,7 +105,7 @@ abstract class BasicAliPay
* @param array|string $options 退款参数或退款商户订单号 * @param array|string $options 退款参数或退款商户订单号
* @param null $refund_amount 退款金额 * @param null $refund_amount 退款金额
* @return array|boolean * @return array|boolean
* @throws \WeChat\Exceptions\InvalidResponseException * @throws InvalidResponseException
*/ */
public function refund($options, $refund_amount = null) public function refund($options, $refund_amount = null)
{ {
@ -117,7 +118,7 @@ abstract class BasicAliPay
* 关闭支付宝进行中的订单 * 关闭支付宝进行中的订单
* @param array|string $options * @param array|string $options
* @return array|boolean * @return array|boolean
* @throws \WeChat\Exceptions\InvalidResponseException * @throws InvalidResponseException
*/ */
public function close($options) public function close($options)
{ {
@ -126,23 +127,42 @@ abstract class BasicAliPay
return $this->getResult($options); return $this->getResult($options);
} }
/**
* 获取通知数据
* @param boolean $needSignType 是否需要sign_type字段
* @return boolean|array
* @throws InvalidResponseException
*/
public function notify($needSignType = false)
{
$data = $_POST;
if (empty($data) || empty($data['sign'])) {
throw new InvalidResponseException('Illegal push request.', 0, $data);
}
$string = $this->getSignContent($data, $needSignType);
$content = wordwrap($this->config->get('public_key'), 64, "\n", true);
$res = "-----BEGIN PUBLIC KEY-----\n{$content}\n-----END PUBLIC KEY-----";
if (openssl_verify($string, base64_decode($data['sign']), $res, OPENSSL_ALGO_SHA256) !== 1) {
throw new InvalidResponseException('Data signature verification failed.', 0, $data);
}
return $data;
}
/** /**
* 验证支付宝支付宝通知 * 验证支付宝支付宝通知
* @param array $data 通知数据 * @param array $data 通知数据
* @param null $sign 数据签名 * @param null|string $sign 数据签名
* @param boolean $sync * @return array|boolean
* @return array|bool * @throws InvalidResponseException
*/ */
public function verify($data, $sign = null, $sync = false) protected function verify($data, $sign)
{ {
if (is_null($this->config->get('public_key'))) {
throw new InvalidArgumentException('Missing Config -- [public_key]');
}
$sign = is_null($sign) ? $data['sign'] : $sign;
$content = wordwrap($this->config->get('public_key'), 64, "\n", true); $content = wordwrap($this->config->get('public_key'), 64, "\n", true);
$string = $sync ? json_encode($data) : $this->getSignContent($data, true);
$res = "-----BEGIN PUBLIC KEY-----\n{$content}\n-----END PUBLIC KEY-----"; $res = "-----BEGIN PUBLIC KEY-----\n{$content}\n-----END PUBLIC KEY-----";
return openssl_verify($string, base64_decode($sign), $res, OPENSSL_ALGO_SHA256) === 1 ? $data : false; if (openssl_verify(json_encode($data, 256), base64_decode($sign), $res, OPENSSL_ALGO_SHA256) !== 1) {
throw new InvalidResponseException('Data signature verification failed.');
}
return $data;
} }
/** /**
@ -151,30 +171,28 @@ abstract class BasicAliPay
*/ */
protected function getSign() protected function getSign()
{ {
if (is_null($this->config->get('private_key'))) {
throw new InvalidArgumentException('Missing Config -- [private_key]');
}
$content = wordwrap($this->config->get('private_key'), 64, "\n", true); $content = wordwrap($this->config->get('private_key'), 64, "\n", true);
$string = "-----BEGIN RSA PRIVATE KEY-----\n{$content}\n-----END RSA PRIVATE KEY-----"; $string = "-----BEGIN RSA PRIVATE KEY-----\n{$content}\n-----END RSA PRIVATE KEY-----";
openssl_sign($this->getSignContent($this->options->get()), $sign, $string, OPENSSL_ALGO_SHA256); openssl_sign($this->getSignContent($this->options->get(), true), $sign, $string, OPENSSL_ALGO_SHA256);
return base64_encode($sign); return base64_encode($sign);
} }
/** /**
* 数据签名处理 * 数据签名处理
* @param array $data * @param array $data 需要进行签名数据
* @param boolean $verify * @param boolean $needSignType 是否需要sign_type字段
* @param array $strs
* @return bool|string * @return bool|string
*/ */
private function getSignContent(array $data, $verify = false, $strs = []) private function getSignContent(array $data, $needSignType = false)
{ {
ksort($data); list($attrs,) = [[], ksort($data)];
foreach ($data as $k => $v) if ($v !== '') { if (isset($data['sign'])) unset($data['sign']);
if ($verify && $k != 'sign' && $k != 'sign_type') array_push($strs, "{$k}={$v}"); if (empty($needSignType)) unset($data['sign_type']);
if (!$verify && $v !== '' && !is_null($v) && $k != 'sign' && '@' != substr($v, 0, 1)) array_push($strs, "{$k}={$v}"); foreach ($data as $key => $value) {
if ($value === '' || is_null($value)) continue;
array_push($attrs, "{$key}={$value}");
} }
return join('&', $strs); return join('&', $attrs);
} }
/** /**
@ -183,15 +201,15 @@ abstract class BasicAliPay
*/ */
protected function applyData($options) protected function applyData($options)
{ {
$this->options['biz_content'] = json_encode($this->params->merge($options), JSON_UNESCAPED_UNICODE); $this->options->set('biz_content', json_encode($this->params->merge($options), 256));
$this->options['sign'] = $this->getSign(); $this->options->set('sign', $this->getSign());
} }
/** /**
* 请求接口并验证访问数据 * 请求接口并验证访问数据
* @param array $options * @param array $options
* @return array|boolean * @return array|boolean
* @throws \WeChat\Exceptions\InvalidResponseException * @throws InvalidResponseException
*/ */
protected function getResult($options) protected function getResult($options)
{ {
@ -199,18 +217,18 @@ abstract class BasicAliPay
$method = str_replace('.', '_', $this->options['method']) . '_response'; $method = str_replace('.', '_', $this->options['method']) . '_response';
$data = json_decode(Tools::get($this->gateway, $this->options->get()), true); $data = json_decode(Tools::get($this->gateway, $this->options->get()), true);
if (!isset($data[$method]['code']) || $data[$method]['code'] !== '10000') { if (!isset($data[$method]['code']) || $data[$method]['code'] !== '10000') {
throw new \WeChat\Exceptions\InvalidResponseException( throw new InvalidResponseException(
"Error: " . "Error: " .
(empty($data[$method]['code']) ? '' : "{$data[$method]['msg']} [{$data[$method]['code']}]\r\n") . (empty($data[$method]['code']) ? '' : "{$data[$method]['msg']} [{$data[$method]['code']}]\r\n") .
(empty($data[$method]['sub_code']) ? '' : "{$data[$method]['sub_msg']} [{$data[$method]['sub_code']}]\r\n"), (empty($data[$method]['sub_code']) ? '' : "{$data[$method]['sub_msg']} [{$data[$method]['sub_code']}]\r\n"),
$data[$method]['code'], $data $data[$method]['code'], $data
); );
} }
return $this->verify($data[$method], $data['sign'], true); return $this->verify($data[$method], $data['sign']);
} }
/** /**
* 生成支付html代码 * 生成支付HTML代码
* @return string * @return string
*/ */
protected function buildPayHtml() protected function buildPayHtml()

View File

@ -22,6 +22,7 @@ try {
// 实例支付对象 // 实例支付对象
$pay = \We::AliPayApp($config); $pay = \We::AliPayApp($config);
// $pay = new \AliPay\App($config); // $pay = new \AliPay\App($config);
// 请参考请求参数https://docs.open.alipay.com/api_1/alipay.trade.app.pay // 请参考请求参数https://docs.open.alipay.com/api_1/alipay.trade.app.pay
$result = $pay->apply([ $result = $pay->apply([
'out_trade_no' => time(), // 商户订单号 'out_trade_no' => time(), // 商户订单号
@ -30,7 +31,7 @@ try {
]); ]);
echo '<pre>'; echo '<pre>';
var_export($result); var_export($result);
} catch (Exception $e) { } catch (\Exception $e) {
echo $e->getMessage(); echo $e->getMessage();
} }

View File

@ -18,20 +18,21 @@ include "../include.php";
// 2. 准备公众号配置参数 // 2. 准备公众号配置参数
$config = include "./alipay.php"; $config = include "./alipay.php";
// 实例支付对象 try {
$pay = \We::AliPayApp($config); // 实例支付对象
// $pay = new \AliPay\App($config); $pay = \We::AliPayApp($config);
if ($pay->verify($_POST)) { // $pay = new \AliPay\App($config);
file_put_contents('notify.txt', "收到来自支付宝的异步通知\r\n", FILE_APPEND); if ($data = $pay->notify()) {
file_put_contents('notify.txt', '订单号:' . $_POST['out_trade_no'] . "\r\n", FILE_APPEND); if (in_array($data['trade_status'], ['TRADE_SUCCESS', 'TRADE_FINISHED'])) {
file_put_contents('notify.txt', '订单金额:' . $_POST['total_amount'] . "\r\n\r\n", FILE_APPEND); // @todo 更新订单状态,支付完成
} else { }
file_put_contents('notify.txt', "收到异步通知\r\n", FILE_APPEND); file_put_contents('notify.txt', "收到来自支付宝的异步通知\r\n", FILE_APPEND);
} file_put_contents('notify.txt', '订单号:' . $data['out_trade_no'] . "\r\n", FILE_APPEND);
file_put_contents('notify.txt', '订单金额:' . $data['total_amount'] . "\r\n\r\n", FILE_APPEND);
// 下面是支付通知处理 } else {
$pay = new \AliPay\App($config); file_put_contents('notify.txt', "收到异步通知\r\n", FILE_APPEND);
$notify = $pay->verify($_POST); }
if (in_array($notify['trade_status'], ['TRADE_SUCCESS', 'TRADE_FINISHED'])) { } catch (\Exception $e) {
// @todo 更新订单状态,支付完成 // 异常处理
echo $e->getMessage();
} }

View File

@ -1,4 +1,17 @@
<?php <?php
// +----------------------------------------------------------------------
// | WeChatDeveloper
// +----------------------------------------------------------------------
// | 版权所有 2014~2018 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
// +----------------------------------------------------------------------
// | 官方网站: http://think.ctolog.com
// +----------------------------------------------------------------------
// | 开源协议 ( https://mit-license.org )
// +----------------------------------------------------------------------
// | github开源项目https://github.com/zoujingli/WeChatDeveloper
// +----------------------------------------------------------------------
return [ return [
// 沙箱模式 // 沙箱模式
'debug' => true, 'debug' => true,