506 lines
17 KiB
PHP
506 lines
17 KiB
PHP
<?php
|
||
|
||
class sandpay_plugin
|
||
{
|
||
static public $info = [
|
||
'name' => 'sandpay', //支付插件英文名称,需和目录名称一致,不能有重复
|
||
'showname' => '杉德支付', //支付插件显示名称
|
||
'author' => '杉德', //支付插件作者
|
||
'link' => 'https://www.sandpay.com.cn/', //支付插件作者链接
|
||
'types' => ['alipay','wxpay','bank'], //支付插件支持的支付方式,可选的有alipay,qqpay,wxpay,bank
|
||
'transtypes' => ['bank'], //支付插件支持的转账方式,可选的有alipay,qqpay,wxpay,bank
|
||
'inputs' => [ //支付插件要求传入的参数以及参数显示名称,可选的有appid,appkey,appsecret,appurl,appmchid
|
||
'appid' => [
|
||
'name' => '商户编号',
|
||
'type' => 'input',
|
||
'note' => '',
|
||
],
|
||
'appkey' => [
|
||
'name' => '私钥证书密码',
|
||
'type' => 'input',
|
||
'note' => '',
|
||
],
|
||
'appswitch' => [
|
||
'name' => '环境选择',
|
||
'type' => 'select',
|
||
'options' => [0=>'生产环境',1=>'测试环境'],
|
||
],
|
||
'product' => [
|
||
'name' => '市场产品',
|
||
'type' => 'select',
|
||
'options' => ['QZF'=>'标准线上收款','CSDB'=>'企业杉德宝'],
|
||
],
|
||
],
|
||
'select_alipay' => [
|
||
'1' => '扫码支付',
|
||
'2' => 'JS支付',
|
||
],
|
||
'select_bank' => [
|
||
'1' => '银联聚合码',
|
||
'2' => '快捷支付',
|
||
],
|
||
'select' => null,
|
||
'note' => '将杉德公钥证书sand.cer、商户私钥证书client.pfx(或商户编号.pfx)上传到 /plugins/sandpay/cert/', //支付密钥填写说明
|
||
'bindwxmp' => true, //是否支持绑定微信公众号
|
||
'bindwxa' => true, //是否支持绑定微信小程序
|
||
];
|
||
|
||
static public function submit(){
|
||
global $siteurl, $channel, $order, $sitename;
|
||
|
||
if($order['typename']=='alipay'){
|
||
if(checkalipay() && in_array('2',$channel['apptype'])){
|
||
return ['type'=>'jump','url'=>'/pay/alipayjs/'.TRADE_NO.'/?d=1'];
|
||
}else{
|
||
return ['type'=>'jump','url'=>'/pay/alipay/'.TRADE_NO.'/'];
|
||
}
|
||
}elseif($order['typename']=='wxpay'){
|
||
if(checkwechat() && $channel['appwxmp']>0){
|
||
return ['type'=>'jump','url'=>'/pay/wxjspay/'.TRADE_NO.'/?d=1'];
|
||
}elseif(checkmobile() && $channel['appwxa']>0){
|
||
return ['type'=>'jump','url'=>'/pay/wxwappay/'.TRADE_NO.'/'];
|
||
}else{
|
||
return ['type'=>'jump','url'=>'/pay/wxpay/'.TRADE_NO.'/'];
|
||
}
|
||
}elseif($order['typename']=='bank'){
|
||
if(in_array('2',$channel['apptype'])){
|
||
return ['type'=>'jump','url'=>'/pay/fastpay/'.TRADE_NO.'/'];
|
||
}else{
|
||
return ['type'=>'jump','url'=>'/pay/bank/'.TRADE_NO.'/'];
|
||
}
|
||
}
|
||
}
|
||
|
||
static public function mapi(){
|
||
global $siteurl, $channel, $order, $conf, $device, $mdevice, $method;
|
||
|
||
if($order['typename']=='alipay'){
|
||
if($mdevice=='alipay' && in_array('2',$channel['apptype'])){
|
||
return ['type'=>'jump','url'=>$siteurl.'pay/alipayjs/'.TRADE_NO.'/?d=1'];
|
||
}else{
|
||
return self::alipay();
|
||
}
|
||
}elseif($order['typename']=='wxpay'){
|
||
if($mdevice=='wechat' && $channel['appwxmp']>0){
|
||
return ['type'=>'jump','url'=>$siteurl.'/pay/wxjspay/'.TRADE_NO.'/?d=1'];
|
||
}elseif($device=='mobile' && $channel['appwxa']>0){
|
||
return ['type'=>'jump','url'=>$siteurl.'/pay/wxwappay/'.TRADE_NO.'/'];
|
||
}else{
|
||
return self::wxpay();
|
||
}
|
||
}elseif($order['typename']=='bank'){
|
||
if(in_array('2',$channel['apptype'])){
|
||
return self::fastpay();
|
||
}else{
|
||
return self::bank();
|
||
}
|
||
}
|
||
}
|
||
|
||
//统一下单
|
||
static private function addOrder($pay_type, $pay_mode, $sub_openid = null, $sub_appid = null){
|
||
global $channel, $order, $ordername, $conf, $clientip, $siteurl;
|
||
|
||
require(PAY_ROOT."inc/SandpayClient.php");
|
||
|
||
$client = new SandpayClient($channel['appid'], $channel['appkey'], $channel['appswitch']);
|
||
$params = [
|
||
'marketProduct' => $channel['product'],
|
||
'outReqTime' => date('YmdHis'),
|
||
'mid' => $channel['appid'],
|
||
'outOrderNo' => TRADE_NO,
|
||
'description' => $ordername,
|
||
'goodsClass' => '01',
|
||
'amount' => $order['realmoney'],
|
||
'payType' => $pay_type,
|
||
'payMode' => $pay_mode,
|
||
'payerInfo' => [
|
||
'payAccLimit' => '',
|
||
],
|
||
'notifyUrl' => $conf['localurl'].'pay/notify/'.TRADE_NO.'/',
|
||
'riskmgtInfo' => [
|
||
'sourceIp' => $clientip,
|
||
],
|
||
];
|
||
if($sub_openid && $sub_appid){
|
||
$params['payerInfo'] = [
|
||
'subAppId' => $sub_appid,
|
||
'subUserId' => $sub_openid,
|
||
'frontUrl' => $siteurl.'pay/return/'.TRADE_NO.'/',
|
||
];
|
||
}elseif($sub_openid){
|
||
$params['payerInfo'] = [
|
||
'userId' => $sub_openid,
|
||
'frontUrl' => $siteurl.'pay/return/'.TRADE_NO.'/',
|
||
];
|
||
}
|
||
|
||
return \lib\Payment::lockPayData(TRADE_NO, function() use($client, $params) {
|
||
global $channel;
|
||
$result = $client->execute('/v4/sd-receipts/api/trans/trans.order.create', $params);
|
||
\lib\Payment::updateOrder(TRADE_NO, $result['sandSerialNo']);
|
||
if($channel['appswitch']==1){
|
||
$log = "商户订单号:".TRADE_NO."\r\n【统一下单接口】请求报文:\r\n".$client->request_body."\r\n【统一下单接口】响应报文:\r\n".$client->response_body."\r\n\r\n";
|
||
file_put_contents(PAY_ROOT.'logs/'.date('Ymd').'.log', $log, FILE_APPEND);
|
||
}
|
||
return $result['credential'];
|
||
});
|
||
}
|
||
|
||
//支付宝扫码支付
|
||
static public function alipay(){
|
||
global $channel, $device, $mdevice, $siteurl;
|
||
if(in_array('2',$channel['apptype']) && !in_array('1',$channel['apptype'])){
|
||
$code_url = $siteurl.'pay/alipayjs/'.TRADE_NO.'/';
|
||
}else{
|
||
try{
|
||
$result = self::addOrder('ALIPAY','QR');
|
||
$code_url = $result['qrCode'];
|
||
}catch(Exception $ex){
|
||
return ['type'=>'error','msg'=>'支付宝支付下单失败!'.$ex->getMessage()];
|
||
}
|
||
}
|
||
|
||
if(checkalipay() || $mdevice=='alipay'){
|
||
return ['type'=>'jump','url'=>$code_url];
|
||
}else{
|
||
return ['type'=>'qrcode','page'=>'alipay_qrcode','url'=>$code_url];
|
||
}
|
||
}
|
||
|
||
static public function alipayjs(){
|
||
[$user_type, $user_id] = alipay_oauth();
|
||
|
||
$blocks = checkBlockUser($user_id, TRADE_NO);
|
||
if($blocks) return $blocks;
|
||
|
||
if($user_type == 'openid'){
|
||
return ['type'=>'error','msg'=>'支付宝快捷登录获取uid失败,需将用户标识切换到uid模式'];
|
||
}
|
||
|
||
try{
|
||
$result = self::addOrder('ALIPAY', 'JSAPI', $user_id);
|
||
}catch(Exception $ex){
|
||
return ['type'=>'error','msg'=>'支付宝支付下单失败!'.$ex->getMessage()];
|
||
}
|
||
|
||
if($_GET['d']=='1'){
|
||
$redirect_url='data.backurl';
|
||
}else{
|
||
$redirect_url='\'/pay/ok/'.TRADE_NO.'/\'';
|
||
}
|
||
return ['type'=>'page','page'=>'alipay_jspay','data'=>['alipay_trade_no'=>$result['tradeNo'], 'redirect_url'=>$redirect_url]];
|
||
}
|
||
|
||
//微信扫码支付
|
||
static public function wxpay(){
|
||
global $channel, $siteurl, $device, $mdevice;
|
||
try{
|
||
$result = self::addOrder('WXPAY','QR');
|
||
$code_url = $result['qrCode'];
|
||
}catch(Exception $ex){
|
||
return ['type'=>'error','msg'=>'微信支付下单失败!'.$ex->getMessage()];
|
||
}
|
||
|
||
if (checkwechat() || $mdevice=='wechat') {
|
||
return ['type'=>'jump','url'=>$code_url];
|
||
} elseif (checkmobile() || $device == 'mobile') {
|
||
return ['type'=>'qrcode','page'=>'wxpay_wap','url'=>$code_url];
|
||
} else {
|
||
return ['type'=>'qrcode','page'=>'wxpay_qrcode','url'=>$code_url];
|
||
}
|
||
}
|
||
|
||
//微信公众号
|
||
static public function wxjspay(){
|
||
global $siteurl, $channel, $order, $ordername, $conf, $clientip;
|
||
|
||
$wxinfo = \lib\Channel::getWeixin($channel['appwxmp']);
|
||
if(!$wxinfo) return ['type'=>'error','msg'=>'支付通道绑定的微信公众号不存在'];
|
||
|
||
try{
|
||
$tools = new \WeChatPay\JsApiTool($wxinfo['appid'], $wxinfo['appsecret']);
|
||
$openid = $tools->GetOpenid();
|
||
}catch(Exception $e){
|
||
return ['type'=>'error','msg'=>$e->getMessage()];
|
||
}
|
||
$blocks = checkBlockUser($openid, TRADE_NO);
|
||
if($blocks) return $blocks;
|
||
|
||
try{
|
||
$payinfo = self::addOrder('WXPAY','JSAPI',$openid,$wxinfo['appid']);
|
||
}catch(Exception $ex){
|
||
return ['type'=>'error','msg'=>'微信支付下单失败!'.$ex->getMessage()];
|
||
}
|
||
|
||
if($_GET['d']==1){
|
||
$redirect_url='data.backurl';
|
||
}else{
|
||
$redirect_url='\'/pay/ok/'.TRADE_NO.'/\'';
|
||
}
|
||
return ['type'=>'page','page'=>'wxpay_jspay','data'=>['jsApiParameters'=>json_encode($payinfo), 'redirect_url'=>$redirect_url]];
|
||
}
|
||
|
||
//微信小程序支付
|
||
static public function wxminipay(){
|
||
global $siteurl, $channel, $order, $ordername, $conf, $clientip;
|
||
|
||
$code = isset($_GET['code'])?trim($_GET['code']):exit('{"code":-1,"msg":"code不能为空"}');
|
||
|
||
//①、获取用户openid
|
||
$wxinfo = \lib\Channel::getWeixin($channel['appwxa']);
|
||
if(!$wxinfo)exit('{"code":-1,"msg":"支付通道绑定的微信小程序不存在"}');
|
||
try{
|
||
$tools = new \WeChatPay\JsApiTool($wxinfo['appid'], $wxinfo['appsecret']);
|
||
$openid = $tools->AppGetOpenid($code);
|
||
}catch(Exception $e){
|
||
exit('{"code":-1,"msg":"'.$e->getMessage().'"}');
|
||
}
|
||
$blocks = checkBlockUser($openid, TRADE_NO);
|
||
if($blocks)exit('{"code":-1,"msg":"'.$blocks['msg'].'"}');
|
||
|
||
//②、统一下单
|
||
try{
|
||
$payinfo = self::addOrder('WXPAY','MINI',$openid,$wxinfo['appid']);
|
||
}catch(Exception $ex){
|
||
exit('{"code":-1,"msg":"微信支付下单失败!'.$ex->getMessage().'"}');
|
||
}
|
||
|
||
exit(json_encode(['code'=>0, 'data'=>json_decode($payinfo, true)]));
|
||
}
|
||
|
||
//微信手机支付
|
||
static public function wxwappay(){
|
||
global $siteurl,$channel, $order, $ordername, $conf, $clientip;
|
||
|
||
$wxinfo = \lib\Channel::getWeixin($channel['appwxa']);
|
||
if(!$wxinfo) return ['type'=>'error','msg'=>'支付通道绑定的微信小程序不存在'];
|
||
try{
|
||
$code_url = wxminipay_jump_scheme($wxinfo['id'], TRADE_NO);
|
||
}catch(Exception $e){
|
||
return ['type'=>'error','msg'=>$e->getMessage()];
|
||
}
|
||
return ['type'=>'scheme','page'=>'wxpay_mini','url'=>$code_url];
|
||
}
|
||
|
||
//云闪付扫码支付
|
||
static public function bank(){
|
||
try{
|
||
$result = self::addOrder('CUPPAY','QR');
|
||
$code_url = $result['qrCode'];
|
||
}catch(Exception $ex){
|
||
return ['type'=>'error','msg'=>'云闪付下单失败!'.$ex->getMessage()];
|
||
}
|
||
|
||
return ['type'=>'qrcode','page'=>'bank_qrcode','url'=>$code_url];
|
||
}
|
||
|
||
//快捷支付
|
||
static public function fastpay(){
|
||
if(!empty($_COOKIE['sandpay_user_id'])){
|
||
$user_id = $_COOKIE['sandpay_user_id'];
|
||
}else{
|
||
$user_id = substr(getSid(), 0, 10);
|
||
setcookie('sandpay_user_id', $user_id, time()+3600*24*365, '/');
|
||
}
|
||
try{
|
||
$result = self::addOrder('FASTPAY','SANDH5',$user_id);
|
||
$jump_url = $result['cashierUrl'];
|
||
}catch(Exception $ex){
|
||
return ['type'=>'error','msg'=>'支付宝支付下单失败!'.$ex->getMessage()];
|
||
}
|
||
return ['type'=>'jump','url'=>$jump_url];
|
||
}
|
||
|
||
//异步回调
|
||
static public function notify(){
|
||
global $channel, $order;
|
||
|
||
$sign = $_POST['sign']; //签名
|
||
$data = $_POST['bizData']; //支付数据
|
||
|
||
require(PAY_ROOT."inc/SandpayClient.php");
|
||
|
||
$client = new SandpayClient($channel['appid'], $channel['appkey'], $channel['appswitch']);
|
||
$verifyFlag = $client->verify($data, $sign);
|
||
|
||
if($verifyFlag){
|
||
$array = json_decode($data, true);
|
||
|
||
if($channel['appswitch']==1){
|
||
$params = [
|
||
'outReqTime' => date('YmdHis'),
|
||
'mid' => $channel['appid'],
|
||
'outOrderNo' => $array['outOrderNo'],
|
||
];
|
||
try{
|
||
$client->execute('/v4/sd-receipts/api/trans/trans.order.query', $params);
|
||
}catch(Exception $e){
|
||
//return ['type'=>'error','msg'=>'订单查询失败 '.$ex->getMessage()];
|
||
}
|
||
$log = "商户订单号:".TRADE_NO."\r\n异步通知:\r\n".$data."\r\n【订单查询接口】请求报文:\r\n".$client->request_body."\r\n【订单查询接口】响应报文:\r\n".$client->response_body."\r\n\r\n";
|
||
file_put_contents(PAY_ROOT.'logs/'.date('Ymd').'.log', $log, FILE_APPEND);
|
||
}
|
||
|
||
if($array['orderStatus'] == 'success'){
|
||
$out_trade_no = $array['outOrderNo'];
|
||
$trade_no = $array['sandSerialNo'];
|
||
$money = $array['amount'];
|
||
$buyer = $array['payer']['payerAccNo'];
|
||
if($out_trade_no == TRADE_NO){
|
||
processNotify($order, $trade_no, $buyer);
|
||
}
|
||
return ['type'=>'html','data'=>'respCode=000000'];
|
||
}
|
||
}
|
||
return ['type'=>'html','data'=>'respCode=020002'];
|
||
}
|
||
|
||
//支付返回页面
|
||
static public function return(){
|
||
return ['type'=>'page','page'=>'return'];
|
||
}
|
||
|
||
//支付成功页面
|
||
static public function ok(){
|
||
return ['type'=>'page','page'=>'ok'];
|
||
}
|
||
|
||
//退款
|
||
static public function refund($order){
|
||
global $channel, $conf;
|
||
if(empty($order))exit();
|
||
|
||
require(PAY_ROOT."inc/SandpayClient.php");
|
||
|
||
$params = [
|
||
'marketProduct' => $channel['product'],
|
||
'outReqTime' => date('YmdHis'),
|
||
'mid' => $channel['appid'],
|
||
'outOrderNo' => $order['refund_no'],
|
||
'oriOutOrderNo' => $order['trade_no'],
|
||
'amount' => $order['refundmoney'],
|
||
'notifyUrl' => $conf['localurl'].'pay/refundnotify/'.TRADE_NO.'/',
|
||
];
|
||
try{
|
||
$client = new SandpayClient($channel['appid'], $channel['appkey'], $channel['appswitch']);
|
||
$result = $client->execute('/v4/sd-receipts/api/trans/trans.order.refund', $params);
|
||
if($channel['appswitch']==1){
|
||
$log = "商户订单号:".$order['trade_no']."\r\n【退货接口】请求报文:\r\n".$client->request_body."\r\n【退货接口】响应报文:\r\n".$client->response_body."\r\n\r\n";
|
||
file_put_contents(PAY_ROOT.'logs/'.date('Ymd').'.log', $log, FILE_APPEND);
|
||
}
|
||
return ['code'=>0, 'trade_no'=>$result['sandSerialNo'], 'refund_fee'=>$result['amount']];
|
||
}catch(Exception $ex){
|
||
return ['code'=>-1,'msg'=>$ex->getMessage()];
|
||
}
|
||
}
|
||
|
||
//退款回调
|
||
static public function refundnotify(){
|
||
global $channel, $order;
|
||
|
||
$sign = $_POST['sign']; //签名
|
||
$data = $_POST['bizData']; //支付数据
|
||
|
||
require(PAY_ROOT."inc/SandpayClient.php");
|
||
|
||
$client = new SandpayClient($channel['appid'], $channel['appkey'], $channel['appswitch']);
|
||
$verifyFlag = $client->verify($data, $sign);
|
||
|
||
if($verifyFlag){
|
||
$array = json_decode($data, true);
|
||
if($array['orderStatus'] == 'success'){
|
||
$out_trade_no = $array['outOrderNo'];
|
||
$trade_no = $array['sandSerialNo'];
|
||
$money = $array['amount'];
|
||
return ['type'=>'html','data'=>'respCode=000000'];
|
||
}
|
||
}
|
||
return ['type'=>'html','data'=>'respCode=020002'];
|
||
}
|
||
|
||
//转账
|
||
static public function transfer($channel, $bizParam){
|
||
if(empty($channel) || empty($bizParam))exit();
|
||
|
||
require(PLUGIN_ROOT.'sandpay/inc/SandpayClient.php');
|
||
|
||
$params = [
|
||
'mid' => $channel['appid'],
|
||
'outOrderNo' => $bizParam['out_biz_no'],
|
||
'amount' => $bizParam['money'],
|
||
'payeeInfo' => [
|
||
'accType' => 'cup',
|
||
'accNo' => $bizParam['payee_account'],
|
||
'accName' => $bizParam['payee_real_name'],
|
||
],
|
||
'payerInfo' => [
|
||
'sdaccSubId' => 'payment',
|
||
'remark' => $bizParam['transfer_desc'],
|
||
],
|
||
];
|
||
|
||
try{
|
||
$client = new SandpayClient($channel['appid'], $channel['appkey'], $channel['appswitch']);
|
||
$result = $client->execute('/v4/sd-payment/api/trans/trans.payment.order.create', $params);
|
||
$status = $result['paymentStatus'] == 'success' ? 1 : 0;
|
||
if($channel['appswitch']==1){
|
||
$log = "商户订单号:".$bizParam['out_biz_no']."\r\n【付款接口】请求报文:\r\n".$client->request_body."\r\n【付款接口】响应报文:\r\n".$client->response_body."\r\n\r\n";
|
||
file_put_contents(PAY_ROOT.'logs/'.date('Ymd').'.log', $log, FILE_APPEND);
|
||
}
|
||
return ['code'=>0, 'status'=>$status, 'orderid'=>$result['sandSerialNo'], 'paydate'=>$result['finishedTime']];
|
||
}catch(Exception $ex){
|
||
return ['code'=>-1, 'msg'=>$ex->getMessage()];
|
||
}
|
||
}
|
||
|
||
//转账查询
|
||
static public function transfer_query($channel, $bizParam){
|
||
if(empty($channel) || empty($bizParam))exit();
|
||
|
||
require(PLUGIN_ROOT.'sandpay/inc/SandpayClient.php');
|
||
|
||
$params = [
|
||
'mid' => $channel['appid'],
|
||
'outReqDate' => substr($bizParam['out_biz_no'], 0, 8),
|
||
'outOrderNo' => $bizParam['out_biz_no'],
|
||
];
|
||
try{
|
||
$client = new SandpayClient($channel['appid'], $channel['appkey'], $channel['appswitch']);
|
||
$result = $client->execute('/v4/sd-payment/api/trans/trans.payment.order.query', $params);
|
||
$status = $result['orderStatus'] == 'success' ? 1 : 0;
|
||
if($channel['appswitch']==1){
|
||
$log = "商户订单号:".$bizParam['out_biz_no']."\r\n【付款订单查询接口】请求报文:\r\n".$client->request_body."\r\n【付款订单查询接口】响应报文:\r\n".$client->response_body."\r\n\r\n";
|
||
file_put_contents(PAY_ROOT.'logs/'.date('Ymd').'.log', $log, FILE_APPEND);
|
||
}
|
||
return ['code'=>0, 'status'=>$status];
|
||
}catch(Exception $ex){
|
||
return ['code'=>-1, 'msg'=>$ex->getMessage()];
|
||
}
|
||
}
|
||
|
||
//余额查询
|
||
static public function balance_query($channel, $bizParam){
|
||
if(empty($channel))exit();
|
||
|
||
require(PLUGIN_ROOT.'sandpay/inc/SandpayClient.php');
|
||
|
||
$params = [
|
||
'mid' => $channel['appid'],
|
||
'sdaccSubId' => 'payment',
|
||
];
|
||
try{
|
||
$client = new SandpayClient($channel['appid'], $channel['appkey'], $channel['appswitch']);
|
||
$result = $client->execute('/v4/sd-payment/api/trans/trans.payment.balance.query', $params);
|
||
$account = $result['accountList'][0];
|
||
if($channel['appswitch']==1){
|
||
$log = "商户订单号:".$bizParam['out_biz_no']."\r\n【余额查询接口】请求报文:\r\n".$client->request_body."\r\n【余额查询接口】响应报文:\r\n".$client->response_body."\r\n\r\n";
|
||
file_put_contents(PAY_ROOT.'logs/'.date('Ymd').'.log', $log, FILE_APPEND);
|
||
}
|
||
if(empty($account)) return ['code'=>-1, 'msg'=>'未查询到账户信息'];
|
||
return ['code'=>0, 'amount'=>$account['availableBal'], 'msg'=>'当前账户可用余额:'.$account['availableBal'].' 元,冻结金额:'.$account['frozenBal'].',在途余额:'.$account['transitBal']];
|
||
}catch(Exception $ex){
|
||
return ['code'=>-1, 'msg'=>$ex->getMessage()];
|
||
}
|
||
}
|
||
} |