<?php
namespace WeChat\Contracts;
use WeChat\Exceptions\InvalidArgumentException;
use WeChat\Exceptions\InvalidResponseException;
* 支付宝支付基类
* @package WeChat\Contracts
*/
abstract class BasicAliPay
{
* 静态缓存
* @var static
*/
protected static $cache;
* 支持配置
* @var DataArray
*/
protected $config;
* 当前请求数据
* @var DataArray
*/
protected $options;
* DzContent数据
* @var DataArray
*/
protected $params;
* 正常请求网关
* @var string
*/
protected $gateway = 'https://openapi.alipay.com/gateway.do?charset=utf-8';
* 构造函数
* @param array $options 必填:appid、public_key、private_key,可选:app_cert、root_cert、cache_path
*/
public function __construct($options)
{
if (empty($options['appid'])) {
throw new InvalidArgumentException('Missing Config -- [appid]');
}
if (empty($options['public_key']) && !empty($options['alipay_cert_path']) && is_file($options['alipay_cert_path'])) {
$options['public_key'] = file_get_contents($options['alipay_cert_path']);
}
if (empty($options['private_key']) && !empty($options['private_key_path']) && is_file($options['private_key_path'])) {
$options['private_key'] = file_get_contents($options['private_key_path']);
}
if (empty($options['public_key'])) {
throw new InvalidArgumentException('Missing Config -- [public_key]');
}
if (empty($options['private_key'])) {
throw new InvalidArgumentException('Missing Config -- [private_key]');
}
if (!empty($options['debug'])) {
$this->gateway = 'https://openapi-sandbox.dl.alipaydev.com/gateway.do?charset=utf-8';
}
$this->params = new DataArray([]);
$this->config = new DataArray($options);
$this->options = new DataArray([
'app_id' => $this->config->get('appid'),
'charset' => empty($options['charset']) ? 'utf-8' : $options['charset'],
'format' => 'JSON',
'version' => '1.0',
'sign_type' => empty($options['sign_type']) ? 'RSA2' : $options['sign_type'],
'timestamp' => date('Y-m-d H:i:s'),
]);
if (isset($options['notify_url']) && $options['notify_url'] !== '') {
$this->options->set('notify_url', $options['notify_url']);
}
if (isset($options['return_url']) && $options['return_url'] !== '') {
$this->options->set('return_url', $options['return_url']);
}
if (isset($options['app_auth_token']) && $options['app_auth_token'] !== '') {
$this->options->set('app_auth_token', $options['app_auth_token']);
}
$appCertPath = $this->config->get('app_cert_path');
$aliRootPath = $this->config->get('alipay_root_path');
if (!$this->config->get('app_cert') && !empty($appCertPath) && is_file($appCertPath)) {
$this->config->set('app_cert', file_get_contents($appCertPath));
}
if (!$this->config->get('root_cert') && !empty($aliRootPath) && is_file($aliRootPath)) {
$this->config->set('root_cert', file_get_contents($aliRootPath));
}
}
* 静态创建对象
* @param array $config
* @return static
*/
public static function instance(array $config)
{
$key = md5(get_called_class() . serialize($config));
if (isset(self::$cache[$key])) return self::$cache[$key];
return self::$cache[$key] = new static($config);
}
* 查询支付宝订单状态
* @param string $outTradeNo 商户订单号
* @return array 订单查询结果
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
*/
public function query($outTradeNo = '')
{
$this->options->set('method', 'alipay.trade.query');
return $this->getResult(['out_trade_no' => $outTradeNo]);
}
* 请求接口并验证访问数据
* @param array $options 请求参数
* @return array 接口响应数据
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
*/
protected function getResult($options)
{
$this->applyData($options);
$method = str_replace('.', '_', $this->options['method']) . '_response';
$data = json_decode(Tools::get($this->gateway, $this->options->get()), true);
if (!isset($data[$method]['code']) || $data[$method]['code'] !== '10000') {
throw new InvalidResponseException(
"Error: " .
(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"),
$data[$method]['code'], $data
);
}
return $data[$method];
}
* 数据包生成及数据签名
* @param array $options
*/
protected function applyData($options)
{
if ($this->config->get('app_cert') && $this->config->get('root_cert')) {
$this->setAppCertSnAndRootCertSn();
}
$this->options->set('biz_content', json_encode($this->params->merge($options), 256));
$this->options->set('sign', $this->getSign());
}
* 新版 设置网关应用公钥证书SN、支付宝根证书SN
*/
protected function setAppCertSnAndRootCertSn()
{
if (!($appCert = $this->config->get('app_cert'))) {
throw new InvalidArgumentException('Missing Config -- [app_cert|app_cert_path]');
}
if (!($rootCert = $this->config->get('root_cert'))) {
throw new InvalidArgumentException('Missing Config -- [root_cert|alipay_root_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 string $sign
* @return string
*/
private function getAppCertSN($sign)
{
$ssl = openssl_x509_parse($sign, true);
$issuer = isset($ssl['issuer']) && is_array($ssl['issuer']) ? $ssl['issuer'] : [];
return md5($this->_arr2str(array_reverse($issuer)) . $ssl['serialNumber']);
}
* 新版 数组转字符串
* @param array $array
* @return string
*/
private function _arr2str($array)
{
$string = [];
if ($array && is_array($array)) {
foreach ($array as $key => $value) {
$string[] = $key . '=' . $value;
}
}
return join(',', $string);
}
* 新版 提取根证书序列号
* @param string $sign
* @return string|null
*/
private function getRootCertSN($sign)
{
if (strlen($sign) < 500 && file_exists($sign)) {
$sign = file_get_contents($sign);
}
$sn = null;
$array = explode('-----END CERTIFICATE-----', $sign);
for ($i = 0; $i < count($array) - 1; $i++) {
$ssl[$i] = openssl_x509_parse($array[$i] . '-----END CERTIFICATE-----', true);
if (strpos($ssl[$i]['serialNumber'], '0x') === 0) {
$ssl[$i]['serialNumber'] = $this->_hex2dec($ssl[$i]['serialNumberHex']);
}
if ($ssl[$i]['signatureTypeLN'] == 'sha1WithRSAEncryption' || $ssl[$i]['signatureTypeLN'] == 'sha256WithRSAEncryption') {
$issuer = isset($ssl[$i]['issuer']) && is_array($ssl[$i]['issuer']) ? $ssl[$i]['issuer'] : [];
if ($sn == null) {
$sn = md5($this->_arr2str(array_reverse($issuer)) . $ssl[$i]['serialNumber']);
} else {
$sn = $sn . '_' . md5($this->_arr2str(array_reverse($issuer)) . $ssl[$i]['serialNumber']);
}
}
}
return $sn;
}
* 新版 0x转高精度数字
* @param string $hex
* @return int|string
*/
private function _hex2dec($hex)
{
list($dec, $len) = [0, strlen($hex)];
for ($i = 1; $i <= $len; $i++) {
$dec = bcadd($dec, bcmul(strval(hexdec($hex[$i - 1])), bcpow('16', strval($len - $i))));
}
return $dec;
}
* 获取数据签名
* @return string
*/
protected function getSign()
{
if ($this->options->get('sign_type') === 'RSA2') {
openssl_sign($this->getSignContent($this->options->get(), true), $sign, $this->getAppPrivateKey(), OPENSSL_ALGO_SHA256);
} else {
openssl_sign($this->getSignContent($this->options->get(), true), $sign, $this->getAppPrivateKey(), OPENSSL_ALGO_SHA1);
}
return base64_encode($sign);
}
* 数据签名处理
* @param array $data 需要进行签名数据
* @param boolean $needSignType 是否需要sign_type字段
* @return string
*/
private function getSignContent(array $data, $needSignType = false)
{
ksort($data);
$attrs = array();
if (isset($data['sign'])) unset($data['sign']);
if (empty($needSignType)) unset($data['sign_type']);
foreach ($data as $key => $value) {
if ($value === '' || is_null($value)) continue;
$attrs[] = "{$key}={$value}";
}
return join('&', $attrs);
}
* 获取应用私钥内容
* @return string
*/
private function getAppPrivateKey()
{
$content = wordwrap($this->trimCert($this->config->get('private_key')), 64, "\n", true);
return "-----BEGIN RSA PRIVATE KEY-----\n{$content}\n-----END RSA PRIVATE KEY-----";
}
* 去除证书前后内容及空白
* @param string $sign
* @return string
*/
protected function trimCert($sign)
{
return preg_replace(['/\s+/', '/-{5}.*?-{5}/'], '', $sign);
}
* 支付宝订单退款操作
* @param array|string $options 退款参数或退款商户订单号
* @param string|null $refundAmount 退款金额
* @return array 退款结果
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
*/
public function refund($options, $refundAmount = null)
{
if (!is_array($options)) $options = ['out_trade_no' => $options, 'refund_amount' => $refundAmount];
$this->options->set('method', 'alipay.trade.refund');
return $this->getResult($options);
}
* 支付宝订单退款查询
* @param array|string $options 退款参数或退款商户订单号
* @param array|null $queryOptions 查询选项
* @return array 退款查询结果
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
*/
public function refundQuery($options, $queryOptions = null)
{
if (!is_array($options)) $options = ['out_trade_no' => $options];
empty($queryOptions) || $options['query_options'] = $queryOptions;
$this->options->set('method', 'alipay.trade.fastpay.refund.query');
return $this->getResult($options);
}
* 关闭支付宝进行中的订单
* @param array|string $options 订单参数或商户订单号
* @return array 关闭结果
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
*/
public function close($options)
{
if (!is_array($options)) $options = ['out_trade_no' => $options];
$this->options->set('method', 'alipay.trade.close');
return $this->getResult($options);
}
* 获取通知数据
*
* @param boolean $needSignType 是否需要sign_type字段
* @param array $parameters
* @return array
* @throws \WeChat\Exceptions\InvalidResponseException
*/
public function notify($needSignType = false, array $parameters = [])
{
$data = empty($parameters) ? $_POST : $parameters;
if (empty($data) || empty($data['sign'])) {
throw new InvalidResponseException('Illegal push request.', 0, $data);
}
$string = $this->getSignContent($data, $needSignType);
if (openssl_verify($string, base64_decode($data['sign']), $this->getAliPublicKey(), OPENSSL_ALGO_SHA256) !== 1) {
throw new InvalidResponseException('Data signature verification failed.', 0, $data);
}
return $data;
}
* 获取支付公钥内容
* @return string
*/
public function getAliPublicKey()
{
$cert = $this->config->get('public_key');
if (strpos(trim($cert), '-----BEGIN CERTIFICATE-----') !== false) {
$pkey = openssl_pkey_get_public($cert);
$keyData = openssl_pkey_get_details($pkey);
return trim($keyData['key']);
} else {
$content = wordwrap($this->trimCert($cert), 64, "\n", true);
return "-----BEGIN PUBLIC KEY-----\n{$content}\n-----END PUBLIC KEY-----";
}
}
* 应用数据操作
* @param array $options
* @return mixed
*/
abstract public function apply($options);
* 通用接口调用(支付宝开放平台)
* @param string $apiMethod API 方法名(如 alipay.trade.query),必填
* @param array|string $data 业务参数,数组会写入 biz_content;字符串尝试解析为 JSON
* @param string $method GET|POST|PUT|DELETE|PATCH,默认 GET(POST 会放入表单)
* @param bool $verify 是否验证返回签名
* @return array 返回解析后的响应体
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
*/
public function callApi($apiMethod, $data = [], $method = 'GET', $verify = false)
{
$method = strtoupper($method);
if (empty($apiMethod)) {
throw new \WeChat\Exceptions\InvalidArgumentException("Missing required parameter -- [apiMethod]");
}
$url = $this->gateway;
$this->options->set('method', $apiMethod);
$params = [];
if (is_array($data)) {
$params = $data;
} elseif (is_string($data) && !empty($data)) {
$decoded = json_decode($data, true);
if (json_last_error() === JSON_ERROR_NONE) {
$params = $decoded;
}
}
if (!empty($params)) {
$this->params->merge($params);
}
if ($this->config->get('app_cert') && $this->config->get('root_cert')) {
$this->setAppCertSnAndRootCertSn();
}
$this->options->set('biz_content', json_encode($this->params->get(), 256));
$this->options->set('sign', $this->getSign());
$methodKey = str_replace('.', '_', $this->options->get('method')) . '_response';
$self = $this;
$processResponse = function ($response) use ($verify, $methodKey, $self) {
$data = json_decode($response, true);
if (json_last_error() !== JSON_ERROR_NONE) {
return $response;
}
if ($verify && isset($data['sign']) && isset($data[$methodKey])) {
return $self->verify($data[$methodKey], $data['sign']);
}
if (isset($data[$methodKey])) {
if (isset($data[$methodKey]['code']) && $data[$methodKey]['code'] !== '10000') {
$msg = isset($data[$methodKey]['msg']) ? "{$data[$methodKey]['msg']} [{$data[$methodKey]['code']}]\r\n" : '';
$subMsg = isset($data[$methodKey]['sub_msg']) ? "{$data[$methodKey]['sub_msg']} [{$data[$methodKey]['sub_code']}]\r\n" : '';
throw new InvalidResponseException("Error: " . $msg . $subMsg, $data[$methodKey]['code'], $data);
}
return $data[$methodKey];
}
return $data;
};
if (in_array($method, ['GET', 'HEAD', 'OPTIONS'])) {
$queryParams = array_merge($this->options->get(), $params);
$response = Tools::get($url, $queryParams);
return $processResponse($response);
}
$postData = array_merge($this->options->get(), $params);
$response = Tools::doRequest($method, $url, ['data' => $postData]);
return $processResponse($response);
}
* 验证接口返回的数据签名
* @param array $data 通知数据
* @param null|string $sign 数据签名
* @return array
* @throws \WeChat\Exceptions\InvalidResponseException
*/
protected function verify($data, $sign)
{
unset($data['sign']);
if ($this->options->get('sign_type') === 'RSA2') {
if (openssl_verify(json_encode($data, 256), base64_decode($sign), $this->getAliPublicKey(), OPENSSL_ALGO_SHA256) !== 1) {
throw new InvalidResponseException('Data signature verification failed by RSA2.');
}
} else {
if (openssl_verify(json_encode($data, 256), base64_decode($sign), $this->getAliPublicKey(), OPENSSL_ALGO_SHA1) !== 1) {
throw new InvalidResponseException('Data signature verification failed by RSA.');
}
}
return $data;
}
* 生成支付HTML代码
* @return string
*/
protected function buildPayHtml()
{
$html = "<form id='alipaysubmit' name='alipaysubmit' action='{$this->gateway}' method='post'>";
foreach ($this->options->get() as $key => $value) {
$value = str_replace("'", ''', $value);
$html .= "<input type='hidden' name='{$key}' value='{$value}'/>";
}
$html .= "<input type='submit' value='ok' style='display:none;'></form>";
return "{$html}<script>document.forms['alipaysubmit'].submit();</script>";
}
}