246 lines
7.7 KiB
PHP
246 lines
7.7 KiB
PHP
<?php
|
|
|
|
require_once 'YsepayResponse.php';
|
|
|
|
/**
|
|
* https://open.ysepay.com/openDocs/summary/
|
|
*/
|
|
class YsepayClient
|
|
{
|
|
|
|
//服务商商户号
|
|
private $partnerId;
|
|
|
|
//银盛公钥证书
|
|
private $businessgateCertPath;
|
|
|
|
//私钥证书文件
|
|
private $privateCertPath;
|
|
|
|
//私钥密码
|
|
private $privateKeyPwd;
|
|
|
|
private $charset = 'UTF-8';
|
|
private $signType = 'RSA';
|
|
|
|
public $notifyUrl;
|
|
public $returnUrl;
|
|
|
|
public function __construct($partnerId, $privateKeyPwd){
|
|
$this->partnerId = $partnerId;
|
|
$this->privateKeyPwd = $privateKeyPwd;
|
|
$this->businessgateCertPath = PAY_ROOT.'cert/businessgate.cer';
|
|
$this->privateCertPath = PAY_ROOT.'cert/client.pfx';
|
|
if(!file_exists($this->businessgateCertPath)){
|
|
throw new \Exception('银盛公钥证书文件businessgate.cer不存在');
|
|
}
|
|
if(!file_exists($this->privateCertPath)){
|
|
throw new \Exception('商户私钥证书文件client.pfx不存在');
|
|
}
|
|
}
|
|
|
|
//扫码支付
|
|
public function execute($method, $bizContent){
|
|
$url = 'https://qrcode.ysepay.com/gateway.do';
|
|
$params = [
|
|
'method' => $method,
|
|
'partner_id' => $this->partnerId,
|
|
'timestamp' => date("Y-m-d H:i:s"),
|
|
'charset' => $this->charset,
|
|
'sign_type' => $this->signType,
|
|
'version' => '3.5'
|
|
];
|
|
if (!empty($this->notifyUrl)) {
|
|
$params["notify_url"] = $this->notifyUrl;
|
|
}
|
|
if (!empty($this->returnUrl)) {
|
|
$params["return_url"] = $this->returnUrl;
|
|
}
|
|
$params['biz_content'] = json_encode($bizContent, JSON_UNESCAPED_SLASHES|JSON_UNESCAPED_UNICODE);
|
|
$params['sign'] = $this->generateSign($params);
|
|
$raw = $this->curl($url, $params);
|
|
$response = new YsepayResponse($raw, $method);
|
|
$this->verifyResponse($response);
|
|
return $response;
|
|
}
|
|
|
|
//页面跳转支付
|
|
public function pageExecute($method, $bizParams){
|
|
$url = 'https://openapi.ysepay.com/gateway.do';
|
|
$params = [
|
|
'method' => $method,
|
|
'partner_id' => $this->partnerId,
|
|
'timestamp' => date("Y-m-d H:i:s"),
|
|
'charset' => $this->charset,
|
|
'sign_type' => $this->signType,
|
|
'version' => '3.0'
|
|
];
|
|
if (!empty($this->notifyUrl)) {
|
|
$params["notify_url"] = $this->notifyUrl;
|
|
}
|
|
if (!empty($this->returnUrl)) {
|
|
$params["return_url"] = $this->returnUrl;
|
|
}
|
|
$params = array_merge($params, $bizParams);
|
|
$params['sign'] = $this->generateSign($params);
|
|
|
|
$html = "<form id='alipaysubmit' name='alipaysubmit' action='{$url}' method='POST'>";
|
|
foreach ($params as $key => $value) {
|
|
if ($this->isEmpty($value)) {
|
|
continue;
|
|
}
|
|
$value = htmlentities($value, ENT_QUOTES | ENT_HTML5);
|
|
$html .= "<input type='hidden' name='{$key}' value='{$value}'/>";
|
|
}
|
|
$html .= "<input type='submit' value='ok' style='display:none;'></form>";
|
|
$html .= "<script>document.forms['alipaysubmit'].submit();</script>";
|
|
|
|
return $html;
|
|
}
|
|
|
|
//异步通知回调验签
|
|
public function verify($params)
|
|
{
|
|
if (!$params || !isset($params['sign'])) {
|
|
return false;
|
|
}
|
|
$sign = $params['sign'];
|
|
$data = $this->getSignContent($params);
|
|
try {
|
|
return $this->rsaPubilcVerify($data, $sign);
|
|
} catch (\Exception $ex) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
//验证返回内容签名
|
|
private function verifyResponse(YsepayResponse $response)
|
|
{
|
|
$signData = $response->getSignData();
|
|
$sign = $response->getSign();
|
|
if ($this->isEmpty($signData) || $this->isEmpty($sign)) {
|
|
throw new \Exception('返回数据解析签名失败');
|
|
}
|
|
$checkResult = $this->rsaPubilcVerify($signData, $sign);
|
|
if (!$checkResult) {
|
|
if (strpos($signData, '\/') > 0) {
|
|
$signData = str_replace('\/', '/', $signData);
|
|
$checkResult = $this->rsaPubilcVerify($signData, $sign);
|
|
}
|
|
if (!$checkResult) {
|
|
throw new \Exception('对返回数据使用银盛公钥验签失败');
|
|
}
|
|
}
|
|
}
|
|
|
|
private function generateSign($params){
|
|
return $this->rsaPrivateSign($this->getSignContent($params));
|
|
}
|
|
|
|
private function getSignContent($params){
|
|
ksort($params);
|
|
unset($params['sign']);
|
|
|
|
$stringToBeSigned = "";
|
|
foreach ($params as $k => $v) {
|
|
if($this->isEmpty($v) || substr($v, 0, 1) == '@') continue;
|
|
$stringToBeSigned .= "&{$k}={$v}";
|
|
}
|
|
$stringToBeSigned = substr($stringToBeSigned, 1);
|
|
return $stringToBeSigned;
|
|
}
|
|
|
|
private function isEmpty($value)
|
|
{
|
|
return $value === null || trim($value) === '';
|
|
}
|
|
|
|
//银盛公钥
|
|
private function getPublicKey()
|
|
{
|
|
$file = file_get_contents($this->businessgateCertPath);
|
|
$cert = chunk_split(base64_encode($file), 64, "\n");
|
|
$cert = "-----BEGIN CERTIFICATE-----\n" . $cert . "-----END CERTIFICATE-----\n";
|
|
$res = openssl_pkey_get_public($cert);
|
|
if (!$res) {
|
|
throw new \Exception('从银盛公钥证书获取公钥失败');
|
|
}
|
|
return $res;
|
|
}
|
|
|
|
//商户私钥
|
|
private function getPrivateKey()
|
|
{
|
|
$file = file_get_contents($this->privateCertPath);
|
|
if (!openssl_pkcs12_read($file, $cert, $this->privateKeyPwd)) {
|
|
throw new \Exception('商户私钥证书解析失败');
|
|
}
|
|
return openssl_pkey_get_private($cert['pkey']);
|
|
}
|
|
|
|
//私钥加签
|
|
private function rsaPrivateSign($data)
|
|
{
|
|
$prikey = $this->getPrivateKey();
|
|
$result = openssl_sign($data, $sign, $prikey);
|
|
if (!$result) throw new \Exception('sign error');
|
|
return base64_encode($sign);
|
|
}
|
|
|
|
//公钥验签
|
|
public function rsaPubilcVerify($data, $sign)
|
|
{
|
|
$pubkey = $this->getPublicKey();
|
|
$result = openssl_verify($data, base64_decode($sign), $pubkey);
|
|
return $result === 1;
|
|
}
|
|
|
|
private function curl($url, $postFields = null)
|
|
{
|
|
$ch = curl_init();
|
|
|
|
curl_setopt($ch, CURLOPT_URL, $url);
|
|
curl_setopt($ch, CURLOPT_FAILONERROR, false);
|
|
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
|
|
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
|
|
curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
|
|
|
|
if (is_array($postFields) && 0 < count($postFields)) {
|
|
$postMultipart = false;
|
|
foreach ($postFields as &$value) {
|
|
if(substr($value, 0, 1) == '@' && class_exists('CURLFile')){
|
|
$postMultipart = true;
|
|
$file = substr($value, 1);
|
|
if(file_exists($file)){
|
|
$value = new \CURLFile($file);
|
|
}
|
|
}
|
|
}
|
|
curl_setopt($ch, CURLOPT_POST, true);
|
|
if($postMultipart){
|
|
curl_setopt($ch, CURLOPT_POSTFIELDS, $postFields);
|
|
}else{
|
|
curl_setopt($ch, CURLOPT_POSTFIELDS, http_build_query($postFields));
|
|
}
|
|
}
|
|
|
|
$response = curl_exec($ch);
|
|
|
|
if (curl_errno($ch) > 0) {
|
|
$errmsg = curl_error($ch);
|
|
curl_close($ch);
|
|
throw new \Exception($errmsg, 0);
|
|
}
|
|
|
|
$httpStatusCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
|
|
if ($httpStatusCode != 200) {
|
|
curl_close($ch);
|
|
throw new \Exception($response, $httpStatusCode);
|
|
}
|
|
|
|
curl_close($ch);
|
|
|
|
return $response;
|
|
}
|
|
|
|
} |